123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465 |
- /*
- * dselect - Debian package maintenance user interface
- * baselist.cc - list of somethings
- *
- * Copyright © 1994,1995 Ian Jackson <ijackson@chiark.greenend.org.uk>
- * Copyright © 2001 Wichert Akkerman <wakkerma@debian.org>
- * Copyright © 2007-2013 Guillem Jover <guillem@debian.org>
- *
- * This 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 2 of the License, or
- * (at your option) any later version.
- *
- * This 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 <https://www.gnu.org/licenses/>.
- */
- #include <config.h>
- #include <compat.h>
- #include <sys/ioctl.h>
- #include <assert.h>
- #include <errno.h>
- #include <string.h>
- #include <termios.h>
- #include <unistd.h>
- #include <stdio.h>
- #include <dpkg/i18n.h>
- #include <dpkg/c-ctype.h>
- #include <dpkg/dpkg.h>
- #include <dpkg/dpkg-db.h>
- #include "dselect.h"
- #include "bindings.h"
- void mywerase(WINDOW *win) {
- int my,mx,y,x;
- getmaxyx(win,my,mx);
- for (y=0; y<my; y++) {
- wmove(win,y,0); for (x=0; x<mx; x++) waddch(win,' ');
- }
- wmove(win,0,0);
- }
- baselist *baselist::signallist = nullptr;
- void baselist::sigwinchhandler(int) {
- int save_errno = errno;
- struct winsize size;
- debug(dbg_general, "baselist::sigwinchhandler(), signallist=%p", signallist);
- baselist *p= signallist;
- p->enddisplay();
- endwin(); initscr();
- if (ioctl(fileno(stdout), TIOCGWINSZ, &size) != 0) ohshite(_("ioctl(TIOCGWINSZ) failed"));
- resizeterm(size.ws_row, size.ws_col); wrefresh(curscr);
- p->startdisplay();
- if (doupdate() == ERR) ohshite(_("doupdate in SIGWINCH handler failed"));
- errno = save_errno;
- }
- static void cu_sigwinch(int, void **argv) {
- struct sigaction *osigactp= (struct sigaction*)argv[0];
- sigset_t *oblockedp= (sigset_t*)argv[1];
- if (sigaction(SIGWINCH, osigactp, nullptr))
- ohshite(_("failed to restore old SIGWINCH sigact"));
- delete osigactp;
- if (sigprocmask(SIG_SETMASK, oblockedp, nullptr))
- ohshite(_("failed to restore old signal mask"));
- delete oblockedp;
- }
- void
- baselist::sigwinch_mask(int how)
- {
- sigset_t sigwinchset;
- sigemptyset(&sigwinchset);
- sigaddset(&sigwinchset,SIGWINCH);
- int rc = sigprocmask(how, &sigwinchset, nullptr);
- if (rc < 0) {
- if (how == SIG_UNBLOCK)
- ohshite(_("failed to unblock SIGWINCH"));
- else
- ohshite(_("failed to block SIGWINCH"));
- }
- }
- void
- baselist::setupsigwinch()
- {
- struct sigaction *osigactp = new(struct sigaction);
- sigset_t *oblockedp = new(sigset_t);
- if (sigprocmask(0, nullptr, oblockedp))
- ohshite(_("failed to get old signal mask"));
- if (sigaction(SIGWINCH, nullptr, osigactp))
- ohshite(_("failed to get old SIGWINCH sigact"));
- push_cleanup(cu_sigwinch, ~0, nullptr, 0, 2, osigactp, oblockedp);
- sigwinch_mask(SIG_BLOCK);
- struct sigaction nsigact;
- memset(&nsigact,0,sizeof(nsigact));
- nsigact.sa_handler= sigwinchhandler;
- sigemptyset(&nsigact.sa_mask);
- //nsigact.sa_flags= SA_INTERRUPT;
- if (sigaction(SIGWINCH, &nsigact, nullptr))
- ohshite(_("failed to set new SIGWINCH sigact"));
- }
- void
- baselist::add_column(column &col, const char *title, int width)
- {
- col.title = title;
- col.x = col_cur_x;
- col.width = width;
- col_cur_x += col.width + gap_width;
- }
- void
- baselist::end_column(column &col, const char *title)
- {
- col.title = title;
- col.x = col_cur_x;
- col.width = total_width - col.x;
- col_cur_x += col.width + gap_width;
- }
- void
- baselist::draw_column_head(column &col)
- {
- mvwaddnstr(colheadspad, 0, col.x, col.title, col.width);
- }
- void
- baselist::draw_column_sep(column &col, int y)
- {
- mvwaddch(listpad, y, col.x - 1, ' ');
- }
- void
- baselist::draw_column_item(column &col, int y, const char *item)
- {
- mvwprintw(listpad, y, col.x, "%-*.*s", col.width, col.width, item);
- }
- void baselist::setheights() {
- int y= ymax - (title_height + colheads_height + thisstate_height);
- assert(y>=1);
- if (showinfo==2 && y>=7) {
- list_height= 5;
- whatinfo_height= 1;
- info_height= y-6;
- } else if (showinfo==1 && y>=10) {
- list_height= y/2;
- info_height= (y-1)/2;
- whatinfo_height= 1;
- } else {
- list_height= y;
- info_height= 0;
- whatinfo_height= 0;
- }
- colheads_row= title_height;
- list_row= colheads_row + colheads_height;
- thisstate_row= list_row + list_height;
- info_row= thisstate_row + thisstate_height;
- whatinfo_row= ymax - 1;
- }
- void baselist::startdisplay() {
- debug(dbg_general, "baselist[%p]::startdisplay()", this);
- cbreak(); noecho(); nonl(); keypad(stdscr,TRUE);
- clear(); wnoutrefresh(stdscr);
- // find attributes
- if (has_colors() && start_color()==OK && COLOR_PAIRS >= numscreenparts) {
- int i;
- printf("allocing\n");
- for (i = 1; i < numscreenparts; i++) {
- if (init_pair(i, color[i].fore, color[i].back) != OK)
- ohshite(_("failed to allocate colour pair"));
- part_attr[i] = COLOR_PAIR(i) | color[i].attr;
- }
- } else {
- /* User defined attributes for B&W mode are not currently supported. */
- part_attr[title] = A_REVERSE;
- part_attr[thisstate] = A_STANDOUT;
- part_attr[list] = 0;
- part_attr[listsel] = A_STANDOUT;
- part_attr[selstate] = A_BOLD;
- part_attr[selstatesel] = A_STANDOUT;
- part_attr[colheads]= A_BOLD;
- part_attr[query] = part_attr[title];
- part_attr[info_body] = part_attr[list];
- part_attr[info_head] = A_BOLD;
- part_attr[whatinfo] = part_attr[thisstate];
- part_attr[helpscreen] = A_NORMAL;
- }
- // set up windows and pads, based on screen size
- getmaxyx(stdscr,ymax,xmax);
- title_height= ymax>=6;
- colheads_height= ymax>=5;
- thisstate_height= ymax>=3;
- setheights();
- setwidths();
- titlewin= newwin(1,xmax, 0,0);
- if (!titlewin) ohshite(_("failed to create title window"));
- wattrset(titlewin, part_attr[title]);
- whatinfowin= newwin(1,xmax, whatinfo_row,0);
- if (!whatinfowin) ohshite(_("failed to create whatinfo window"));
- wattrset(whatinfowin, part_attr[whatinfo]);
- listpad = newpad(ymax, total_width);
- if (!listpad) ohshite(_("failed to create baselist pad"));
- colheadspad= newpad(1, total_width);
- if (!colheadspad) ohshite(_("failed to create heading pad"));
- wattrset(colheadspad, part_attr[colheads]);
- thisstatepad= newpad(1, total_width);
- if (!thisstatepad) ohshite(_("failed to create thisstate pad"));
- wattrset(thisstatepad, part_attr[thisstate]);
- infopad= newpad(MAX_DISPLAY_INFO, total_width);
- if (!infopad) ohshite(_("failed to create info pad"));
- wattrset(infopad, part_attr[info_body]);
- wbkgdset(infopad, ' ' | part_attr[info_body]);
- querywin= newwin(1,xmax,ymax-1,0);
- if (!querywin) ohshite(_("failed to create query window"));
- wbkgdset(querywin, ' ' | part_attr[query]);
- if (cursorline >= topofscreen + list_height) topofscreen= cursorline;
- if (topofscreen > nitems - list_height) topofscreen= nitems - list_height;
- if (topofscreen < 0) topofscreen= 0;
- infotopofscreen= 0; leftofscreen= 0;
- redrawall();
- debug(dbg_general,
- "baselist::startdisplay() done ...\n\n"
- " xmax=%d, ymax=%d;\n\n"
- " title_height=%d, colheads_height=%d, list_height=%d;\n"
- " thisstate_height=%d, info_height=%d, whatinfo_height=%d;\n\n"
- " colheads_row=%d, thisstate_row=%d, info_row=%d;\n"
- " whatinfo_row=%d, list_row=%d;\n\n",
- xmax, ymax, title_height, colheads_height, list_height,
- thisstate_height, info_height, whatinfo_height,
- colheads_row, thisstate_row, info_row, whatinfo_row, list_row);
- }
- void baselist::enddisplay() {
- delwin(titlewin);
- delwin(whatinfowin);
- delwin(listpad);
- delwin(colheadspad);
- delwin(thisstatepad);
- delwin(infopad);
- wmove(stdscr,ymax,0); wclrtoeol(stdscr);
- listpad = nullptr;
- col_cur_x = 0;
- }
- void baselist::redrawall() {
- redrawtitle();
- redrawcolheads();
- wattrset(listpad, part_attr[list]);
- mywerase(listpad);
- ldrawnstart= ldrawnend= -1; // start is first drawn; end is first undrawn; -1=none
- refreshlist();
- redrawthisstate();
- redrawinfo();
- }
- void baselist::redraw1item(int index) {
- redraw1itemsel(index, index == cursorline);
- }
- baselist::baselist(keybindings *kb) {
- debug(dbg_general, "baselist[%p]::baselist()", this);
- bindings= kb;
- nitems= 0;
- col_cur_x = 0;
- gap_width = 1;
- total_width = max(TOTAL_LIST_WIDTH, COLS);
- xmax= -1;
- ymax = -1;
- list_height = 0;
- info_height = 0;
- title_height = 0;
- whatinfo_height = 0;
- colheads_height = 0;
- thisstate_height = 0;
- list_row = 0;
- info_row = 0;
- whatinfo_row = 0;
- colheads_row = 0;
- thisstate_row = 0;
- topofscreen = 0;
- leftofscreen = 0;
- infotopofscreen = 0;
- infolines = 0;
- listpad = nullptr;
- infopad = nullptr;
- colheadspad = nullptr;
- thisstatepad = nullptr;
- titlewin = nullptr;
- querywin = nullptr;
- whatinfowin = nullptr;
- cursorline = -1;
- ldrawnstart = 0;
- ldrawnend = 0;
- showinfo= 1;
- searchstring[0]= 0;
- }
- void baselist::itd_keys() {
- whatinfovb(_("Keybindings"));
- const int givek= xmax/3;
- bindings->describestart();
- const char **ta;
- while ((ta = bindings->describenext()) != nullptr) {
- const char **tap= ta+1;
- for (;;) {
- waddstr(infopad, gettext(*tap));
- tap++; if (!*tap) break;
- waddstr(infopad, ", ");
- }
- int y,x;
- getyx(infopad,y,x);
- if (x >= givek) y++;
- mvwaddstr(infopad, y,givek, ta[0]);
- waddch(infopad,'\n');
- delete [] ta;
- }
- }
- void baselist::dosearch() {
- int offset, index;
- debug(dbg_general, "baselist[%p]::dosearch(); searchstring='%s'",
- this, searchstring);
- for (offset = 1, index = max(topofscreen, cursorline + 1);
- offset<nitems;
- offset++, index++) {
- if (index >= nitems) index -= nitems;
- if (matchsearch(index)) {
- topofscreen= index-1;
- if (topofscreen > nitems - list_height) topofscreen= nitems-list_height;
- if (topofscreen < 0) topofscreen= 0;
- setcursor(index);
- return;
- }
- }
- beep();
- }
- void baselist::refreshinfo() {
- pnoutrefresh(infopad, infotopofscreen,leftofscreen, info_row,0,
- min(info_row + info_height - 1, info_row + MAX_DISPLAY_INFO - 1),
- min(total_width - leftofscreen - 1, xmax - 1));
- if (whatinfo_height) {
- mywerase(whatinfowin);
- mvwaddstr(whatinfowin,0,0, whatinfovb.string());
- if (infolines > info_height) {
- wprintw(whatinfowin,_(" -- %d%%, press "),
- (int)((infotopofscreen + info_height) * 100.0 / infolines));
- if (infotopofscreen + info_height < infolines) {
- wprintw(whatinfowin,_("%s for more"), bindings->find("iscrollon"));
- if (infotopofscreen) waddstr(whatinfowin, ", ");
- }
- if (infotopofscreen)
- wprintw(whatinfowin, _("%s to go back"),bindings->find("iscrollback"));
- waddch(whatinfowin,'.');
- }
- wnoutrefresh(whatinfowin);
- }
- }
- void baselist::wordwrapinfo(int offset, const char *m) {
- int usemax= xmax-5;
- debug(dbg_general, "baselist[%p]::wordwrapinfo(%d, '%s')", this, offset, m);
- bool wrapping = false;
- for (;;) {
- int offleft=offset; while (*m == ' ' && offleft>0) { m++; offleft--; }
- const char *p= strchr(m,'\n');
- int l= p ? (int)(p-m) : strlen(m);
- while (l && c_isspace(m[l - 1]))
- l--;
- if (!l || (*m == '.' && l == 1)) {
- if (wrapping) waddch(infopad,'\n');
- waddch(infopad, '\n');
- wrapping = false;
- } else if (*m == ' ' || usemax < 10) {
- if (wrapping) waddch(infopad,'\n');
- waddnstr(infopad, m, l);
- waddch(infopad, '\n');
- wrapping = false;
- } else {
- int x, y DPKG_ATTR_UNUSED;
- if (wrapping) {
- getyx(infopad, y,x);
- if (x+1 >= usemax) {
- waddch(infopad,'\n');
- } else {
- waddch(infopad,' ');
- }
- }
- for (;;) {
- getyx(infopad, y,x);
- int dosend= usemax-x;
- if (l <= dosend) {
- dosend=l;
- } else {
- int i=dosend;
- while (i > 0 && m[i] != ' ') i--;
- if (i > 0 || x > 0) dosend=i;
- }
- if (dosend) waddnstr(infopad, m, dosend);
- while (dosend < l && m[dosend] == ' ') dosend++;
- l-= dosend; m+= dosend;
- if (l <= 0) break;
- waddch(infopad,'\n');
- }
- wrapping = true;
- }
- if (!p) break;
- if (getcury(infopad) == (MAX_DISPLAY_INFO - 1)) {
- waddstr(infopad,
- "[The package description is too long and has been truncated...]");
- break;
- }
- m= ++p;
- }
- debug(dbg_general, "baselist[%p]::wordwrapinfo() done", this);
- }
- baselist::~baselist() { }
|