+int
+swap_menu(BoardMenu* boardmenu, int src_index, int dest_index)
+{
+ /* reposition menus */
+ mvwin(get_menu_win(boardmenu->menu_list[src_index]),
+ 1, 1+MENU_WIDTH*dest_index
+ );
+ mvwin(get_menu_win(boardmenu->menu_list[dest_index]),
+ 1, 1+MENU_WIDTH*src_index
+ );
+ refresh();
+ wrefresh(get_menu_win(boardmenu->menu_list[src_index]));
+ wrefresh(get_menu_win(boardmenu->menu_list[dest_index]));
+ /* wclear(get_menu_win(boardmenu->menu_list[src_index])); */
+ /* wclear(get_menu_win(boardmenu->menu_list[dest_index])); */
+ /* touchwin(get_menu_win(boardmenu->menu_list[src_index])); */
+ /* touchwin(get_menu_win(boardmenu->menu_list[dest_index])); */
+ clear();
+
+ /* swap in array */
+ ar_swap_item((void*)boardmenu->menu_list, src_index, dest_index);
+
+ return 0;
+}
+
+/* menuitem */
+int
+update_menuitem_descrip(MenuItem* menuitem)
+{ /* need to do something about colored text */
+
+ TodoItem* item_data;
+ char* new_descrip;
+
+ item_data = (TodoItem*)get_menuitem_userdata(menuitem);
+ new_descrip = strdup("");
+
+ if (strlen(item_data->description) > 0) {
+ /* strcat(new_descrip, "☰ "); */
+ strcat(new_descrip, "# ");
+ }
+ if (strlen(item_data->due) > 0) {
+ strcat(new_descrip, item_data->due);
+ strcat(new_descrip, " ");
+ }
+ if (item_data->subtask_count > 0) {
+
+ int tasks_complete = 0;
+ for (int i = 0; i < item_data->subtask_count; i++) {
+ if (item_data->subtask_list[i]->done == SubTaskState_done) {
+ tasks_complete += 1;
+ }
+ }
+
+ /* [, # done, /, # total, ], null */
+ char subtask_done[4]; // assume there wont be more than 999 subtasks (possibly danger?)
+ snprintf(subtask_done, 4, "%d", tasks_complete);
+ int substask_len = 1+item_data->subtask_count+1+strlen(subtask_done)+1+1;
+ char subtask_text[substask_len];
+ sprintf(subtask_text, "[%s/%d]", subtask_done, item_data->subtask_count);
+ strcat(new_descrip, subtask_text);
+ }
+
+ /* free old string */
+ if (strlen(new_descrip) > 0) {
+ free(get_menuitem_descrip(menuitem));
+ set_menuitem_descrip(menuitem, new_descrip);
+ }
+
+ return 0;
+}
+
+/* popup */
+PopupMenu*
+make_popupmenu(TodoItem* itemdata)
+{
+ PopupMenu* new_popupmenu;
+ MenuItem** subtask_menuitems;
+ Menu* popupmenu_menu;
+ WINDOW* popupmenu_win;
+ WINDOW* popupmenu_menu_win;
+
+ new_popupmenu = malloc(sizeof(PopupMenu));
+
+ subtask_menuitems = subtasklist_to_menuitem(itemdata->subtask_list, itemdata->subtask_count);
+ popupmenu_menu = create_menu(strdup(""), subtask_menuitems);
+
+ /* popup win */
+ int maxheight, maxwidth;
+ getmaxyx(stdscr, maxheight, maxwidth);
+ popupmenu_win = newwin(
+ maxheight-2*POPUP_BORDER,
+ maxwidth-2*2*POPUP_BORDER,
+ POPUP_BORDER,
+ POPUP_BORDER*2
+ );
+
+ int popup_maxheight, popup_maxwidth;
+ getmaxyx(popupmenu_win, popup_maxheight, popup_maxwidth);
+ popupmenu_menu_win = derwin(
+ popupmenu_win,
+ popup_maxheight-POPUP_MENU_PAD_TOP-POPUP_MENU_PAD_BOTTOM,
+ popup_maxwidth-POPUP_MENU_PAD_LEFT-POPUP_MENU_PAD_RIGHT,
+ POPUP_MENU_PAD_TOP,
+ POPUP_MENU_PAD_LEFT
+ );
+
+ set_menu_win(popupmenu_menu, popupmenu_menu_win);
+ set_menu_focus(popupmenu_menu, 1);
+ set_menu_renderitem(popupmenu_menu, render_popup_menuitem);
+ set_menu_itemheight(popupmenu_menu, popup_menuitem_height);
+ box(popupmenu_win, 0, 0);
+
+ /* move this stuff to render phase later? */
+ mvwprintw(popupmenu_win, 1, 2, itemdata->item_name);
+ mvwprintw(popupmenu_win, 2, 2, (strlen(itemdata->description) > 0) ? itemdata->description : "no description");
+
+ /* don't forget to free popupmenu after */
+ new_popupmenu->win = popupmenu_win;
+ new_popupmenu->menu = popupmenu_menu;
+
+ return new_popupmenu;
+}
+
+void
+render_popup_menuitem(Menu* menu, int item_index, int start_y)
+{
+ MenuItem* curitem;
+ WINDOW* menu_win;
+ int hlcolor;
+
+ curitem = get_menu_item(menu, item_index);
+ menu_win = get_menu_win(menu);
+
+ /* color selected item */
+ hlcolor = COLOR_PAIR((item_index == get_selected_item(menu) && get_menu_focused(menu) == true) ? TS_SELECTED : TS_NONSELECTED);
+ wattron(menu_win, hlcolor);
+
+ wmove(menu_win, start_y, 0);
+ /* print subtask done indicator */
+ if (strlen(get_menuitem_title(curitem)) > 0)
+ wprintw(
+ menu_win,
+ (((SubTask*)get_menuitem_userdata(curitem))->done == SubTaskState_done) ? "[X] " : "[ ] "
+ );
+ wprintw(menu_win, get_menuitem_title(curitem));
+
+ wattroff(menu_win, hlcolor);
+}
+
+int
+popup_menuitem_height(MenuItem* menuitem)
+{
+ return 1; // account for wrap later
+}
+
+int
+render_popupmenu(PopupMenu* popupmenu)
+{
+ render_menu(popupmenu->menu);
+ wrefresh(popupmenu->win);
+
+ return 0;
+}
+
+/* this is copy paste of other, prob abstract */
+MenuItem**
+subtasklist_to_menuitem(SubTask** subtask_list, int list_length)
+{
+ MenuItem** items;
+
+ items = malloc((list_length+1)*sizeof(MenuItem*));
+ for (int i = 0; i < list_length; i++) {
+ MenuItem* new_menuitem;
+ new_menuitem = create_menuitem(subtask_list[i]->subtask_name);
+ /* using same struct, careful if it gets freed */
+ set_menuitem_userdata(new_menuitem, subtask_list[i]);
+
+ items[i] = new_menuitem;
+ }
+
+ items[list_length] = 0; //null terminate
+ return items;
+}
+
+/* helpers */
+int
+ungetstr(char* str)
+{
+ // ignore null character (it's fine even if strlen = 0)
+ for (int i = strlen(str)-1; i >= 0; i--) {
+ ungetch(str[i]);
+ }
+
+ return 0;
+}
+