Move widget add/del API from WDialog to WGroup.
[midnight-commander.git] / src / filemanager / chown.c
blobabfb41506323f945b17504fe6611a088a2e5a3dc
1 /*
2 Chown command -- for the Midnight Commander
4 Copyright (C) 1994-2020
5 Free Software Foundation, Inc.
7 This file is part of the Midnight Commander.
9 The Midnight Commander is free software: you can redistribute it
10 and/or modify it under the terms of the GNU General Public License as
11 published by the Free Software Foundation, either version 3 of the License,
12 or (at your option) any later version.
14 The Midnight Commander is distributed in the hope that it will be useful,
15 but WITHOUT ANY WARRANTY; without even the implied warranty of
16 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
17 GNU General Public License for more details.
19 You should have received a copy of the GNU General Public License
20 along with this program. If not, see <http://www.gnu.org/licenses/>.
23 /** \file chown.c
24 * \brief Source: chown command
27 #include <config.h>
29 #include <errno.h>
30 #include <stdio.h>
31 #include <string.h>
32 #include <pwd.h>
33 #include <grp.h>
34 #include <sys/types.h>
35 #include <sys/stat.h>
36 #include <unistd.h>
38 #include "lib/global.h"
40 #include "lib/tty/tty.h"
41 #include "lib/skin.h"
42 #include "lib/vfs/vfs.h"
43 #include "lib/strutil.h"
44 #include "lib/util.h"
45 #include "lib/widget.h"
47 #include "src/setup.h" /* panels_options */
48 #include "midnight.h" /* current_panel */
50 #include "chown.h"
52 /*** global variables ****************************************************************************/
54 /*** file scope macro definitions ****************************************************************/
56 #define GH 12
57 #define GW 21
59 #define BUTTONS 5
61 #define B_SETALL B_USER
62 #define B_SETUSR (B_USER + 1)
63 #define B_SETGRP (B_USER + 2)
65 #define LABELS 5
67 #define chown_label(n,txt) label_set_text (chown_label [n].l, txt)
69 /*** file scope type declarations ****************************************************************/
71 /*** file scope variables ************************************************************************/
74 static struct
76 int ret_cmd;
77 button_flags_t flags;
78 int y;
79 int len;
80 const char *text;
81 } chown_but[BUTTONS] =
83 /* *INDENT-OFF* */
84 { B_SETALL, NORMAL_BUTTON, 5, 0, N_("Set &all") },
85 { B_SETGRP, NORMAL_BUTTON, 5, 0, N_("Set &groups") },
86 { B_SETUSR, NORMAL_BUTTON, 5, 0, N_("Set &users") },
87 { B_ENTER, DEFPUSH_BUTTON, 3, 0, N_("&Set") },
88 { B_CANCEL, NORMAL_BUTTON, 3, 0, N_("&Cancel") }
89 /* *INDENT-ON* */
92 /* summary length of three buttons */
93 static int blen = 0;
95 static struct
97 int y;
98 WLabel *l;
99 } chown_label[LABELS] =
101 /* *INDENT-OFF* */
102 { 4, NULL },
103 { 6, NULL },
104 { 8, NULL },
105 { 10, NULL },
106 { 12, NULL }
107 /* *INDENT-ON* */
110 static int current_file;
111 static gboolean ignore_all;
113 static WListbox *l_user, *l_group;
115 /* --------------------------------------------------------------------------------------------- */
116 /*** file scope functions ************************************************************************/
117 /* --------------------------------------------------------------------------------------------- */
119 static void
120 chown_i18n (void)
122 static gboolean i18n = FALSE;
123 int i;
125 if (i18n)
126 return;
128 i18n = TRUE;
130 #ifdef ENABLE_NLS
131 for (i = 0; i < BUTTONS; i++)
132 chown_but[i].text = _(chown_but[i].text);
133 #endif /* ENABLE_NLS */
135 for (i = 0; i < BUTTONS; i++)
137 chown_but[i].len = str_term_width1 (chown_but[i].text) + 3; /* [], spaces and w/o & */
138 if (chown_but[i].flags == DEFPUSH_BUTTON)
139 chown_but[i].len += 2; /* <> */
141 if (i < BUTTONS - 2)
142 blen += chown_but[i].len;
145 blen += 2;
148 /* --------------------------------------------------------------------------------------------- */
150 static void
151 chown_refresh (WDialog * h)
153 int y = 3;
154 int x = 7 + GW * 2;
156 dlg_default_repaint (h);
158 tty_setcolor (COLOR_NORMAL);
160 widget_gotoyx (h, y + 0, x);
161 tty_print_string (_("Name"));
162 widget_gotoyx (h, y + 2, x);
163 tty_print_string (_("Owner name"));
164 widget_gotoyx (h, y + 4, x);
165 tty_print_string (_("Group name"));
166 widget_gotoyx (h, y + 6, x);
167 tty_print_string (_("Size"));
168 widget_gotoyx (h, y + 8, x);
169 tty_print_string (_("Permission"));
172 /* --------------------------------------------------------------------------------------------- */
174 static cb_ret_t
175 chown_callback (Widget * w, Widget * sender, widget_msg_t msg, int parm, void *data)
177 switch (msg)
179 case MSG_DRAW:
180 chown_refresh (DIALOG (w));
181 return MSG_HANDLED;
183 default:
184 return dlg_default_callback (w, sender, msg, parm, data);
188 /* --------------------------------------------------------------------------------------------- */
190 static WDialog *
191 chown_init (void)
193 int single_set;
194 WDialog *ch_dlg;
195 WGroup *g;
196 int lines, cols;
197 int i, y;
198 struct passwd *l_pass;
199 struct group *l_grp;
201 single_set = (current_panel->marked < 2) ? 3 : 0;
202 lines = GH + 4 + (single_set != 0 ? 2 : 4);
203 cols = GW * 3 + 2 + 6;
205 ch_dlg =
206 dlg_create (TRUE, 0, 0, lines, cols, WPOS_CENTER, FALSE, dialog_colors, chown_callback,
207 NULL, "[Chown]", _("Chown command"));
208 g = GROUP (ch_dlg);
210 group_add_widget (g, groupbox_new (2, 3, GH, GW, _("User name")));
211 l_user = listbox_new (3, 4, GH - 2, GW - 2, FALSE, NULL);
212 group_add_widget (g, l_user);
213 /* add field for unknown names (numbers) */
214 listbox_add_item (l_user, LISTBOX_APPEND_AT_END, 0, _("<Unknown user>"), NULL, FALSE);
215 /* get and put user names in the listbox */
216 setpwent ();
217 while ((l_pass = getpwent ()) != NULL)
218 listbox_add_item (l_user, LISTBOX_APPEND_SORTED, 0, l_pass->pw_name, NULL, FALSE);
219 endpwent ();
221 group_add_widget (g, groupbox_new (2, 4 + GW, GH, GW, _("Group name")));
222 l_group = listbox_new (3, 5 + GW, GH - 2, GW - 2, FALSE, NULL);
223 group_add_widget (g, l_group);
224 /* add field for unknown names (numbers) */
225 listbox_add_item (l_group, LISTBOX_APPEND_AT_END, 0, _("<Unknown group>"), NULL, FALSE);
226 /* get and put group names in the listbox */
227 setgrent ();
228 while ((l_grp = getgrent ()) != NULL)
229 listbox_add_item (l_group, LISTBOX_APPEND_SORTED, 0, l_grp->gr_name, NULL, FALSE);
230 endgrent ();
232 group_add_widget (g, groupbox_new (2, 5 + GW * 2, GH, GW, _("File")));
233 /* add widgets for the file information */
234 for (i = 0; i < LABELS; i++)
236 chown_label[i].l = label_new (chown_label[i].y, 7 + GW * 2, "");
237 group_add_widget (g, chown_label[i].l);
240 if (single_set == 0)
242 int x;
244 group_add_widget (g, hline_new (lines - chown_but[0].y - 1, -1, -1));
246 y = lines - chown_but[0].y;
247 x = (cols - blen) / 2;
249 for (i = 0; i < BUTTONS - 2; i++)
251 group_add_widget (g, button_new (y, x, chown_but[i].ret_cmd, chown_but[i].flags,
252 chown_but[i].text, NULL));
253 x += chown_but[i].len + 1;
257 i = BUTTONS - 2;
258 y = lines - chown_but[i].y;
259 group_add_widget (g, hline_new (y - 1, -1, -1));
260 group_add_widget (g, button_new (y, WIDGET (ch_dlg)->cols / 2 - chown_but[i].len,
261 chown_but[i].ret_cmd, chown_but[i].flags, chown_but[i].text,
262 NULL));
263 i++;
264 group_add_widget (g, button_new (y, WIDGET (ch_dlg)->cols / 2 + 1, chown_but[i].ret_cmd,
265 chown_but[i].flags, chown_but[i].text, NULL));
267 /* select first listbox */
268 widget_select (WIDGET (l_user));
270 return ch_dlg;
273 /* --------------------------------------------------------------------------------------------- */
275 static void
276 chown_done (gboolean need_update)
278 if (need_update)
279 update_panels (UP_OPTIMIZE, UP_KEEPSEL);
280 repaint_screen ();
283 /* --------------------------------------------------------------------------------------------- */
285 static const char *
286 next_file (void)
288 while (!current_panel->dir.list[current_file].f.marked)
289 current_file++;
291 return current_panel->dir.list[current_file].fname;
294 /* --------------------------------------------------------------------------------------------- */
296 static gboolean
297 try_chown (const vfs_path_t * p, uid_t u, gid_t g)
299 while (mc_chown (p, u, g) == -1 && !ignore_all)
301 int my_errno = errno;
302 int result;
303 char *msg;
305 msg =
306 g_strdup_printf (_("Cannot chown \"%s\"\n%s"), x_basename (vfs_path_as_str (p)),
307 unix_error_string (my_errno));
308 result =
309 query_dialog (MSG_ERROR, msg, D_ERROR, 4, _("&Ignore"), _("Ignore &all"), _("&Retry"),
310 _("&Cancel"));
311 g_free (msg);
313 switch (result)
315 case 0:
316 /* try next file */
317 return TRUE;
319 case 1:
320 ignore_all = TRUE;
321 /* try next file */
322 return TRUE;
324 case 2:
325 /* retry this file */
326 break;
328 case 3:
329 default:
330 /* stop remain files processing */
331 return FALSE;
335 return TRUE;
338 /* --------------------------------------------------------------------------------------------- */
340 static gboolean
341 do_chown (const vfs_path_t * p, uid_t u, gid_t g)
343 gboolean ret;
345 ret = try_chown (p, u, g);
347 do_file_mark (current_panel, current_file, 0);
349 return ret;
352 /* --------------------------------------------------------------------------------------------- */
354 static void
355 apply_chowns (vfs_path_t * vpath, uid_t u, gid_t g)
357 gboolean ok;
359 if (!do_chown (vpath, u, g))
360 return;
364 const char *fname;
365 struct stat sf;
367 fname = next_file ();
368 vpath = vfs_path_from_str (fname);
369 ok = (mc_stat (vpath, &sf) == 0);
371 if (!ok)
373 /* if current file was deleted outside mc -- try next file */
374 /* decrease current_panel->marked */
375 do_file_mark (current_panel, current_file, 0);
377 /* try next file */
378 ok = TRUE;
380 else
381 ok = do_chown (vpath, u, g);
383 vfs_path_free (vpath);
385 while (ok && current_panel->marked != 0);
388 /* --------------------------------------------------------------------------------------------- */
389 /*** public functions ****************************************************************************/
390 /* --------------------------------------------------------------------------------------------- */
392 void
393 chown_cmd (void)
395 gboolean need_update;
396 gboolean end_chown;
398 chown_i18n ();
400 current_file = 0;
401 ignore_all = FALSE;
404 { /* do while any files remaining */
405 vfs_path_t *vpath;
406 WDialog *ch_dlg;
407 struct stat sf_stat;
408 const char *fname;
409 int result;
410 char buffer[BUF_TINY];
411 uid_t new_user = (uid_t) (-1);
412 gid_t new_group = (gid_t) (-1);
414 do_refresh ();
416 need_update = FALSE;
417 end_chown = FALSE;
419 if (current_panel->marked != 0)
420 fname = next_file (); /* next marked file */
421 else
422 fname = selection (current_panel)->fname; /* single file */
424 vpath = vfs_path_from_str (fname);
426 if (mc_stat (vpath, &sf_stat) != 0)
428 vfs_path_free (vpath);
429 break;
432 ch_dlg = chown_init ();
434 /* select in listboxes */
435 listbox_select_entry (l_user, listbox_search_text (l_user, get_owner (sf_stat.st_uid)));
436 listbox_select_entry (l_group, listbox_search_text (l_group, get_group (sf_stat.st_gid)));
438 chown_label (0, str_trunc (fname, GW - 4));
439 chown_label (1, str_trunc (get_owner (sf_stat.st_uid), GW - 4));
440 chown_label (2, str_trunc (get_group (sf_stat.st_gid), GW - 4));
441 size_trunc_len (buffer, GW - 4, sf_stat.st_size, 0, panels_options.kilobyte_si);
442 chown_label (3, buffer);
443 chown_label (4, string_perm (sf_stat.st_mode));
445 result = dlg_run (ch_dlg);
447 switch (result)
449 case B_CANCEL:
450 end_chown = TRUE;
451 break;
453 case B_ENTER:
454 case B_SETALL:
456 struct group *grp;
457 struct passwd *user;
458 char *text;
460 listbox_get_current (l_group, &text, NULL);
461 grp = getgrnam (text);
462 if (grp != NULL)
463 new_group = grp->gr_gid;
464 listbox_get_current (l_user, &text, NULL);
465 user = getpwnam (text);
466 if (user != NULL)
467 new_user = user->pw_uid;
468 if (result == B_ENTER)
470 if (current_panel->marked <= 1)
472 /* single or last file */
473 if (mc_chown (vpath, new_user, new_group) == -1)
474 message (D_ERROR, MSG_ERROR, _("Cannot chown \"%s\"\n%s"),
475 fname, unix_error_string (errno));
476 end_chown = TRUE;
478 else if (!try_chown (vpath, new_user, new_group))
480 /* stop multiple files processing */
481 result = B_CANCEL;
482 end_chown = TRUE;
485 else
487 apply_chowns (vpath, new_user, new_group);
488 end_chown = TRUE;
491 need_update = TRUE;
492 break;
495 case B_SETUSR:
497 struct passwd *user;
498 char *text;
500 listbox_get_current (l_user, &text, NULL);
501 user = getpwnam (text);
502 if (user != NULL)
504 new_user = user->pw_uid;
505 apply_chowns (vpath, new_user, new_group);
506 need_update = TRUE;
507 end_chown = TRUE;
509 break;
512 case B_SETGRP:
514 struct group *grp;
515 char *text;
517 listbox_get_current (l_group, &text, NULL);
518 grp = getgrnam (text);
519 if (grp != NULL)
521 new_group = grp->gr_gid;
522 apply_chowns (vpath, new_user, new_group);
523 need_update = TRUE;
524 end_chown = TRUE;
526 break;
529 default:
530 break;
533 if (current_panel->marked != 0 && result != B_CANCEL)
535 do_file_mark (current_panel, current_file, 0);
536 need_update = TRUE;
539 vfs_path_free (vpath);
541 dlg_destroy (ch_dlg);
543 while (current_panel->marked != 0 && !end_chown);
545 chown_done (need_update);
548 /* --------------------------------------------------------------------------------------------- */