chiark / gitweb /
dpkg (1.18.25) stretch; urgency=medium
[dpkg] / dselect / pkgtop.cc
1 /*
2  * dselect - Debian package maintenance user interface
3  * pkgtop.cc - handles (re)draw of package list windows colheads, list, thisstate
4  *
5  * Copyright © 1994,1995 Ian Jackson <ijackson@chiark.greenend.org.uk>
6  * Copyright © 2007-2014 Guillem Jover <guillem@debian.org>
7  *
8  * This is free software; you can redistribute it and/or modify
9  * it under the terms of the GNU General Public License as published by
10  * the Free Software Foundation; either version 2 of the License, or
11  * (at your option) any later version.
12  *
13  * This is distributed in the hope that it will be useful,
14  * but WITHOUT ANY WARRANTY; without even the implied warranty of
15  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
16  * GNU General Public License for more details.
17  *
18  * You should have received a copy of the GNU General Public License
19  * along with this program.  If not, see <https://www.gnu.org/licenses/>.
20  */
21
22 #include <config.h>
23 #include <compat.h>
24
25 #include <assert.h>
26 #include <string.h>
27 #include <stdio.h>
28
29 #include <dpkg/i18n.h>
30 #include <dpkg/c-ctype.h>
31 #include <dpkg/dpkg.h>
32 #include <dpkg/dpkg-db.h>
33
34 #include "dselect.h"
35 #include "pkglist.h"
36
37 static const char *
38 pkgprioritystring(const struct pkginfo *pkg)
39 {
40   if (pkg->priority == PKG_PRIO_UNSET) {
41     return nullptr;
42   } else if (pkg->priority == PKG_PRIO_OTHER) {
43     return pkg->otherpriority;
44   } else {
45     assert(pkg->priority <= PKG_PRIO_UNKNOWN);
46     return gettext(prioritystrings[pkg->priority]);
47   }
48 }
49
50 int packagelist::describemany(char buf[], const char *prioritystring,
51                               const char *section,
52                               const struct perpackagestate *pps) {
53   const char *ssostring, *ssoabbrev;
54   int statindent;
55
56   statindent= 0;
57   ssostring = nullptr;
58   ssoabbrev= _("All");
59   switch (statsortorder) {
60   case sso_avail:
61     if (pps->ssavail == -1) break;
62     ssostring= ssastrings[pps->ssavail];
63     ssoabbrev= ssaabbrevs[pps->ssavail];
64     statindent++;
65     break;
66   case sso_state:
67     if (pps->ssstate == -1) break;
68     ssostring= sssstrings[pps->ssstate];
69     ssoabbrev= sssabbrevs[pps->ssstate];
70     statindent++;
71     break;
72   case sso_unsorted:
73     break;
74   default:
75     internerr("unknown statsortrder %d", statsortorder);
76   }
77
78   if (!prioritystring) {
79     if (!section) {
80       strcpy(buf, ssostring ? gettext(ssostring) : _("All packages"));
81       return statindent;
82     } else {
83       if (!*section) {
84         sprintf(buf,_("%s packages without a section"),gettext(ssoabbrev));
85       } else {
86         sprintf(buf,_("%s packages in section %s"),gettext(ssoabbrev),section);
87       }
88       return statindent+1;
89     }
90   } else {
91     if (!section) {
92       sprintf(buf,_("%s %s packages"),gettext(ssoabbrev),prioritystring);
93       return statindent+1;
94     } else {
95       if (!*section) {
96         sprintf(buf,_("%s %s packages without a section"),gettext(ssoabbrev),prioritystring);
97       } else {
98         sprintf(buf,_("%s %s packages in section %s"),gettext(ssoabbrev),prioritystring,section);
99       }
100       return statindent+2;
101     }
102   }
103 }
104
105 void packagelist::redrawthisstate() {
106   if (!thisstate_height) return;
107   mywerase(thisstatepad);
108
109   const char *section= table[cursorline]->pkg->section;
110   const char *priority= pkgprioritystring(table[cursorline]->pkg);
111   char *buf= new char[500+
112                       max((table[cursorline]->pkg->set->name ?
113                            strlen(table[cursorline]->pkg->set->name) : 0),
114                           (section ? strlen(section) : 0) +
115                           (priority ? strlen(priority) : 0))];
116
117   if (table[cursorline]->pkg->set->name) {
118     sprintf(buf,
119             _("%-*s %s%s%s;  %s (was: %s).  %s"),
120             col_package.width,
121             table[cursorline]->pkg->set->name,
122             gettext(statusstrings[table[cursorline]->pkg->status]),
123             ((eflagstrings[table[cursorline]->pkg->eflag][0]==' ') &&
124               (eflagstrings[table[cursorline]->pkg->eflag][1]=='\0'))  ? "" : " - ",
125             gettext(eflagstrings[table[cursorline]->pkg->eflag]),
126             gettext(wantstrings[table[cursorline]->selected]),
127             gettext(wantstrings[table[cursorline]->original]),
128             priority);
129   } else {
130     describemany(buf,priority,section,table[cursorline]->pkg->clientdata);
131   }
132   mvwaddnstr(thisstatepad,0,0, buf, total_width);
133   pnoutrefresh(thisstatepad, 0,leftofscreen, thisstate_row,0,
134                thisstate_row, min(total_width - 1, xmax - 1));
135
136   delete[] buf;
137 }
138
139 void packagelist::redraw1itemsel(int index, int selected) {
140   int i, indent, j;
141   const char *p;
142   const struct pkginfo *pkg= table[index]->pkg;
143   int screenline = index - topofscreen;
144
145   wattrset(listpad, part_attr[selected ? listsel : list]);
146
147   if (pkg->set->name) {
148     if (verbose) {
149       draw_column_item(col_status_hold, screenline,
150                        gettext(eflagstrings[pkg->eflag]));
151
152       draw_column_sep(col_status_status, screenline);
153       draw_column_item(col_status_status, screenline,
154                        gettext(statusstrings[pkg->status]));
155
156       draw_column_sep(col_status_old_want, screenline);
157       draw_column_item(col_status_old_want, screenline,
158               /* FIXME: keep this? */
159               /*table[index]->original == table[index]->selected ? "(same)"
160               : */gettext(wantstrings[table[index]->original]));
161
162       draw_column_sep(col_status_new_want, screenline);
163       wattrset(listpad, part_attr[selected ? selstatesel : selstate]);
164       draw_column_item(col_status_new_want, screenline,
165                        gettext(wantstrings[table[index]->selected]));
166
167       wattrset(listpad, part_attr[selected ? listsel : list]);
168
169       draw_column_sep(col_priority, screenline);
170       draw_column_item(col_priority, screenline,
171                        pkg->priority == PKG_PRIO_OTHER ?
172                        pkg->otherpriority :
173                        gettext(prioritystrings[pkg->priority]));
174     } else {
175       mvwaddch(listpad, screenline, 0, eflagchars[pkg->eflag]);
176       waddch(listpad, statuschars[pkg->status]);
177       waddch(listpad,
178              /* FIXME: keep this feature? */
179              /*table[index]->original == table[index]->selected ? ' '
180              : */wantchars[table[index]->original]);
181
182       wattrset(listpad, part_attr[selected ? selstatesel : selstate]);
183       waddch(listpad, wantchars[table[index]->selected]);
184       wattrset(listpad, part_attr[selected ? listsel : list]);
185
186       wmove(listpad, screenline, col_priority.x - 1);
187       waddch(listpad, ' ');
188       if (pkg->priority == PKG_PRIO_OTHER) {
189         for (i = col_priority.width, p = pkg->otherpriority;
190              i > 0 && *p;
191              i--, p++)
192           waddch(listpad, c_tolower(*p));
193         while (i-- > 0) waddch(listpad,' ');
194       } else {
195         wprintw(listpad, "%-*.*s", col_priority.width, col_priority.width,
196                 gettext(priorityabbrevs[pkg->priority]));
197       }
198     }
199
200     draw_column_sep(col_section, screenline);
201     draw_column_item(col_section, screenline,
202                      pkg->section ? pkg->section : "?");
203
204     draw_column_sep(col_package, screenline);
205     draw_column_item(col_package, screenline,
206                      pkg->set->name);
207
208     waddch(listpad, ' ');
209
210     if (col_archinstalled.width) {
211       draw_column_sep(col_archinstalled, screenline);
212       draw_column_item(col_archinstalled, screenline, pkg->installed.arch->name);
213
214       waddch(listpad, ' ');
215     }
216     if (col_archavailable.width) {
217       draw_column_sep(col_archavailable, screenline);
218       draw_column_item(col_archavailable, screenline, pkg->available.arch->name);
219
220       waddch(listpad, ' ');
221     }
222
223     if (col_versioninstalled.width) {
224       draw_column_item(col_versioninstalled, screenline,
225                        versiondescribe(&pkg->installed.version, vdew_nonambig));
226       waddch(listpad, ' ');
227     }
228     if (col_versionavailable.width) {
229       if (dpkg_version_is_informative(&pkg->available.version) &&
230           dpkg_version_compare(&pkg->available.version,
231                                &pkg->installed.version) > 0)
232         wattrset(listpad, part_attr[selected ? selstatesel : selstate]);
233       draw_column_item(col_versionavailable, screenline,
234                        versiondescribe(&pkg->available.version, vdew_nonambig));
235       wattrset(listpad, part_attr[selected ? listsel : list]);
236       waddch(listpad,' ');
237     }
238
239     i = col_description.width;
240     p = pkg->available.description ? pkg->available.description :
241         pkg->installed.description ? pkg->installed.description : "";
242     while (i>0 && *p && *p != '\n') { waddnstr(listpad,p,1); i--; p++; }
243   } else {
244     const char *section= pkg->section;
245     const char *priority= pkgprioritystring(pkg);
246
247     char *buf= new char[500+
248                         (section ? strlen(section) : 0) +
249                         (priority ? strlen(priority) : 0)];
250
251     indent= describemany(buf,priority,section,pkg->clientdata);
252
253     mvwaddstr(listpad, screenline, 0, "    ");
254     i= total_width-7;
255     j= (indent<<1) + 1;
256     while (j-- >0) { waddch(listpad,ACS_HLINE); i--; }
257     waddch(listpad,' ');
258
259     wattrset(listpad, part_attr[selected ? selstatesel : selstate]);
260     p= buf;
261     while (i>0 && *p) { waddnstr(listpad, p,1); p++; i--; }
262     wattrset(listpad, part_attr[selected ? listsel : list]);
263
264     waddch(listpad,' ');
265     j= (indent<<1) + 1;
266     while (j-- >0) { waddch(listpad,ACS_HLINE); i--; }
267
268     delete[] buf;
269   }
270
271   while (i>0) { waddch(listpad,' '); i--; }
272 }
273
274 void packagelist::redrawcolheads() {
275   if (colheads_height) {
276     wattrset(colheadspad, part_attr[colheads]);
277     mywerase(colheadspad);
278     if (verbose) {
279       wmove(colheadspad,0,0);
280       for (int i = 0; i < col_status_old_want.width; i++)
281         waddch(colheadspad, '.');
282       draw_column_head(col_status_hold);
283       draw_column_head(col_status_status);
284       draw_column_head(col_status_old_want);
285       draw_column_head(col_status_new_want);
286     } else {
287       draw_column_head(col_status);
288     }
289
290     draw_column_head(col_section);
291     draw_column_head(col_priority);
292     draw_column_head(col_package);
293
294     if (col_archinstalled.width)
295       draw_column_head(col_archinstalled);
296     if (col_archavailable.width)
297       draw_column_head(col_archavailable);
298
299     if (col_versioninstalled.width)
300       draw_column_head(col_versioninstalled);
301     if (col_versionavailable.width)
302       draw_column_head(col_versionavailable);
303
304     draw_column_head(col_description);
305   }
306   refreshcolheads();
307 }