xref: /freebsd/contrib/ncurses/menu/m_item_new.c (revision 21817992b3314c908ab50f0bb88d2ee750b9c4ac)
10e3d5408SPeter Wemm /****************************************************************************
2*21817992SBaptiste Daroussin  * Copyright 2020-2021 Thomas E. Dickey                                     *
3e1865124SBaptiste Daroussin  * Copyright 1998-2010,2012 Free Software Foundation, Inc.                  *
40e3d5408SPeter Wemm  *                                                                          *
50e3d5408SPeter Wemm  * Permission is hereby granted, free of charge, to any person obtaining a  *
60e3d5408SPeter Wemm  * copy of this software and associated documentation files (the            *
70e3d5408SPeter Wemm  * "Software"), to deal in the Software without restriction, including      *
80e3d5408SPeter Wemm  * without limitation the rights to use, copy, modify, merge, publish,      *
90e3d5408SPeter Wemm  * distribute, distribute with modifications, sublicense, and/or sell       *
100e3d5408SPeter Wemm  * copies of the Software, and to permit persons to whom the Software is    *
110e3d5408SPeter Wemm  * furnished to do so, subject to the following conditions:                 *
120e3d5408SPeter Wemm  *                                                                          *
130e3d5408SPeter Wemm  * The above copyright notice and this permission notice shall be included  *
140e3d5408SPeter Wemm  * in all copies or substantial portions of the Software.                   *
150e3d5408SPeter Wemm  *                                                                          *
160e3d5408SPeter Wemm  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS  *
170e3d5408SPeter Wemm  * OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF               *
180e3d5408SPeter Wemm  * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.   *
190e3d5408SPeter Wemm  * IN NO EVENT SHALL THE ABOVE COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM,   *
200e3d5408SPeter Wemm  * DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR    *
210e3d5408SPeter Wemm  * OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR    *
220e3d5408SPeter Wemm  * THE USE OR OTHER DEALINGS IN THE SOFTWARE.                               *
230e3d5408SPeter Wemm  *                                                                          *
240e3d5408SPeter Wemm  * Except as contained in this notice, the name(s) of the above copyright   *
250e3d5408SPeter Wemm  * holders shall not be used in advertising or otherwise to promote the     *
260e3d5408SPeter Wemm  * sale, use or other dealings in this Software without prior written       *
270e3d5408SPeter Wemm  * authorization.                                                           *
280e3d5408SPeter Wemm  ****************************************************************************/
290e3d5408SPeter Wemm 
300e3d5408SPeter Wemm /****************************************************************************
314a1a9510SRong-En Fan  *   Author:  Juergen Pfeifer, 1995,1997                                    *
320e3d5408SPeter Wemm  ****************************************************************************/
330e3d5408SPeter Wemm 
340e3d5408SPeter Wemm /***************************************************************************
350e3d5408SPeter Wemm * Module m_item_new                                                        *
360e3d5408SPeter Wemm * Create and destroy menu items                                            *
370e3d5408SPeter Wemm * Set and get marker string for menu                                       *
380e3d5408SPeter Wemm ***************************************************************************/
390e3d5408SPeter Wemm 
400e3d5408SPeter Wemm #include "menu.priv.h"
410e3d5408SPeter Wemm 
424a1a9510SRong-En Fan #if USE_WIDEC_SUPPORT
434a1a9510SRong-En Fan #if HAVE_WCTYPE_H
444a1a9510SRong-En Fan #include <wctype.h>
454a1a9510SRong-En Fan #endif
464a1a9510SRong-En Fan #endif
474a1a9510SRong-En Fan 
48*21817992SBaptiste Daroussin MODULE_ID("$Id: m_item_new.c,v 1.38 2021/06/17 21:26:02 tom Exp $")
490e3d5408SPeter Wemm 
500e3d5408SPeter Wemm /*---------------------------------------------------------------------------
510e3d5408SPeter Wemm |   Facility      :  libnmenu
520e3d5408SPeter Wemm |   Function      :  bool Is_Printable_String(const char *s)
530e3d5408SPeter Wemm |
540e3d5408SPeter Wemm |   Description   :  Checks whether or not the string contains only printable
550e3d5408SPeter Wemm |                    characters.
560e3d5408SPeter Wemm |
570e3d5408SPeter Wemm |   Return Values :  TRUE     - if string is printable
580e3d5408SPeter Wemm |                    FALSE    - if string contains non-printable characters
590e3d5408SPeter Wemm +--------------------------------------------------------------------------*/
604a1a9510SRong-En Fan static bool
Is_Printable_String(const char * s)614a1a9510SRong-En Fan Is_Printable_String(const char *s)
620e3d5408SPeter Wemm {
634a1a9510SRong-En Fan   int result = TRUE;
644a1a9510SRong-En Fan 
654a1a9510SRong-En Fan #if USE_WIDEC_SUPPORT
6673f0a83dSXin LI   int count = (int)mbstowcs(0, s, 0);
674a1a9510SRong-En Fan   wchar_t *temp = 0;
684a1a9510SRong-En Fan 
694a1a9510SRong-En Fan   assert(s);
704a1a9510SRong-En Fan 
714a1a9510SRong-En Fan   if (count > 0
724a1a9510SRong-En Fan       && (temp = typeCalloc(wchar_t, (2 + (unsigned)count))) != 0)
734a1a9510SRong-En Fan     {
744a1a9510SRong-En Fan       int n;
754a1a9510SRong-En Fan 
764a1a9510SRong-En Fan       mbstowcs(temp, s, (unsigned)count);
774a1a9510SRong-En Fan       for (n = 0; n < count; ++n)
784a1a9510SRong-En Fan 	if (!iswprint((wint_t)temp[n]))
794a1a9510SRong-En Fan 	  {
804a1a9510SRong-En Fan 	    result = FALSE;
814a1a9510SRong-En Fan 	    break;
824a1a9510SRong-En Fan 	  }
834a1a9510SRong-En Fan       free(temp);
844a1a9510SRong-En Fan     }
854a1a9510SRong-En Fan #else
860e3d5408SPeter Wemm   assert(s);
870e3d5408SPeter Wemm   while (*s)
880e3d5408SPeter Wemm     {
894a1a9510SRong-En Fan       if (!isprint(UChar(*s)))
904a1a9510SRong-En Fan 	{
914a1a9510SRong-En Fan 	  result = FALSE;
924a1a9510SRong-En Fan 	  break;
934a1a9510SRong-En Fan 	}
940e3d5408SPeter Wemm       s++;
950e3d5408SPeter Wemm     }
964a1a9510SRong-En Fan #endif
974a1a9510SRong-En Fan   return result;
980e3d5408SPeter Wemm }
990e3d5408SPeter Wemm 
1000e3d5408SPeter Wemm /*---------------------------------------------------------------------------
1010e3d5408SPeter Wemm |   Facility      :  libnmenu
1020e3d5408SPeter Wemm |   Function      :  ITEM *new_item(char *name, char *description)
1030e3d5408SPeter Wemm |
1040e3d5408SPeter Wemm |   Description   :  Create a new item with name and description. Return
1050e3d5408SPeter Wemm |                    a pointer to this new item.
1060e3d5408SPeter Wemm |                    N.B.: an item must(!) have a name.
1070e3d5408SPeter Wemm |
1080e3d5408SPeter Wemm |   Return Values :  The item pointer or NULL if creation failed.
1090e3d5408SPeter Wemm +--------------------------------------------------------------------------*/
1107a656419SBaptiste Daroussin MENU_EXPORT(ITEM *)
new_item(const char * name,const char * description)1117a69bbfbSPeter Wemm new_item(const char *name, const char *description)
1120e3d5408SPeter Wemm {
1130e3d5408SPeter Wemm   ITEM *item;
1140e3d5408SPeter Wemm 
1154a1a9510SRong-En Fan   T((T_CALLED("new_item(\"%s\", \"%s\")"),
1164a1a9510SRong-En Fan      name ? name : "",
1174a1a9510SRong-En Fan      description ? description : ""));
1184a1a9510SRong-En Fan 
1190e3d5408SPeter Wemm   if (!name || (*name == '\0') || !Is_Printable_String(name))
1200e3d5408SPeter Wemm     {
1210e3d5408SPeter Wemm       item = (ITEM *)0;
1220e3d5408SPeter Wemm       SET_ERROR(E_BAD_ARGUMENT);
1230e3d5408SPeter Wemm     }
1240e3d5408SPeter Wemm   else
1250e3d5408SPeter Wemm     {
12606bfebdeSXin LI       item = typeCalloc(ITEM, 1);
1277a656419SBaptiste Daroussin 
1280e3d5408SPeter Wemm       if (item)
1290e3d5408SPeter Wemm 	{
1307a656419SBaptiste Daroussin 	  T((T_CREATE("item %p"), (void *)item));
1310e3d5408SPeter Wemm 	  *item = _nc_Default_Item;	/* hope we have struct assignment */
1320e3d5408SPeter Wemm 
13373f0a83dSXin LI 	  item->name.length = (unsigned short)strlen(name);
1340e3d5408SPeter Wemm 	  item->name.str = name;
1350e3d5408SPeter Wemm 
1360e3d5408SPeter Wemm 	  if (description && (*description != '\0') &&
1370e3d5408SPeter Wemm 	      Is_Printable_String(description))
1380e3d5408SPeter Wemm 	    {
13973f0a83dSXin LI 	      item->description.length = (unsigned short)strlen(description);
1400e3d5408SPeter Wemm 	      item->description.str = description;
1410e3d5408SPeter Wemm 	    }
1420e3d5408SPeter Wemm 	  else
1430e3d5408SPeter Wemm 	    {
1440e3d5408SPeter Wemm 	      item->description.length = 0;
1450e3d5408SPeter Wemm 	      item->description.str = (char *)0;
1460e3d5408SPeter Wemm 	    }
1470e3d5408SPeter Wemm 	}
1480e3d5408SPeter Wemm       else
1490e3d5408SPeter Wemm 	SET_ERROR(E_SYSTEM_ERROR);
1500e3d5408SPeter Wemm     }
1514a1a9510SRong-En Fan   returnItem(item);
1520e3d5408SPeter Wemm }
1530e3d5408SPeter Wemm 
1540e3d5408SPeter Wemm /*---------------------------------------------------------------------------
1550e3d5408SPeter Wemm |   Facility      :  libnmenu
1560e3d5408SPeter Wemm |   Function      :  int free_item(ITEM *item)
1570e3d5408SPeter Wemm |
1580e3d5408SPeter Wemm |   Description   :  Free the allocated storage for this item.
1590e3d5408SPeter Wemm |                    N.B.: a connected item can't be freed.
1600e3d5408SPeter Wemm |
1610e3d5408SPeter Wemm |   Return Values :  E_OK              - success
1620e3d5408SPeter Wemm |                    E_BAD_ARGUMENT    - invalid value has been passed
1630e3d5408SPeter Wemm |                    E_CONNECTED       - item is still connected to a menu
1640e3d5408SPeter Wemm +--------------------------------------------------------------------------*/
1657a656419SBaptiste Daroussin MENU_EXPORT(int)
free_item(ITEM * item)1667a69bbfbSPeter Wemm free_item(ITEM *item)
1670e3d5408SPeter Wemm {
16806bfebdeSXin LI   T((T_CALLED("free_item(%p)"), (void *)item));
1694a1a9510SRong-En Fan 
1700e3d5408SPeter Wemm   if (!item)
1710e3d5408SPeter Wemm     RETURN(E_BAD_ARGUMENT);
1720e3d5408SPeter Wemm 
1730e3d5408SPeter Wemm   if (item->imenu)
1740e3d5408SPeter Wemm     RETURN(E_CONNECTED);
1750e3d5408SPeter Wemm 
1760e3d5408SPeter Wemm   free(item);
1770e3d5408SPeter Wemm 
1780e3d5408SPeter Wemm   RETURN(E_OK);
1790e3d5408SPeter Wemm }
1800e3d5408SPeter Wemm 
1810e3d5408SPeter Wemm /*---------------------------------------------------------------------------
1820e3d5408SPeter Wemm |   Facility      :  libnmenu
1830e3d5408SPeter Wemm |   Function      :  int set_menu_mark( MENU *menu, const char *mark )
1840e3d5408SPeter Wemm |
1850e3d5408SPeter Wemm |   Description   :  Set the mark string used to indicate the current
1860e3d5408SPeter Wemm |                    item (single-valued menu) or the selected items
1870e3d5408SPeter Wemm |                    (multi-valued menu).
1880e3d5408SPeter Wemm |                    The mark argument may be NULL, in which case no
1890e3d5408SPeter Wemm |                    marker is used.
1900e3d5408SPeter Wemm |                    This might be a little bit tricky, because this may
1910e3d5408SPeter Wemm |                    affect the geometry of the menu, which we don't allow
1920e3d5408SPeter Wemm |                    if it is already posted.
1930e3d5408SPeter Wemm |
1940e3d5408SPeter Wemm |   Return Values :  E_OK               - success
1950e3d5408SPeter Wemm |                    E_BAD_ARGUMENT     - an invalid value has been passed
1960e3d5408SPeter Wemm |                    E_SYSTEM_ERROR     - no memory to store mark
1970e3d5408SPeter Wemm +--------------------------------------------------------------------------*/
1987a656419SBaptiste Daroussin MENU_EXPORT(int)
set_menu_mark(MENU * menu,const char * mark)1997a69bbfbSPeter Wemm set_menu_mark(MENU *menu, const char *mark)
2000e3d5408SPeter Wemm {
20173f0a83dSXin LI   short l;
2024a1a9510SRong-En Fan 
20306bfebdeSXin LI   T((T_CALLED("set_menu_mark(%p,%s)"), (void *)menu, _nc_visbuf(mark)));
2040e3d5408SPeter Wemm 
2050e3d5408SPeter Wemm   if (mark && (*mark != '\0') && Is_Printable_String(mark))
20673f0a83dSXin LI     l = (short)strlen(mark);
2070e3d5408SPeter Wemm   else
2080e3d5408SPeter Wemm     l = 0;
2090e3d5408SPeter Wemm 
2100e3d5408SPeter Wemm   if (menu)
2110e3d5408SPeter Wemm     {
2120e3d5408SPeter Wemm       char *old_mark = menu->mark;
2130e3d5408SPeter Wemm       unsigned short old_status = menu->status;
2140e3d5408SPeter Wemm 
2150e3d5408SPeter Wemm       if (menu->status & _POSTED)
2160e3d5408SPeter Wemm 	{
2170e3d5408SPeter Wemm 	  /* If the menu is already posted, the geometry is fixed. Then
2180e3d5408SPeter Wemm 	     we can only accept a mark with exactly the same length */
21973f0a83dSXin LI 	  if (menu->marklen != l)
2200e3d5408SPeter Wemm 	    RETURN(E_BAD_ARGUMENT);
2210e3d5408SPeter Wemm 	}
2220e3d5408SPeter Wemm       menu->marklen = l;
2230e3d5408SPeter Wemm       if (l)
2240e3d5408SPeter Wemm 	{
22506bfebdeSXin LI 	  menu->mark = strdup(mark);
2260e3d5408SPeter Wemm 	  if (menu->mark)
2270e3d5408SPeter Wemm 	    {
2280e3d5408SPeter Wemm 	      if (menu != &_nc_Default_Menu)
22973f0a83dSXin LI 		SetStatus(menu, _MARK_ALLOCATED);
2300e3d5408SPeter Wemm 	    }
2310e3d5408SPeter Wemm 	  else
2320e3d5408SPeter Wemm 	    {
2330e3d5408SPeter Wemm 	      menu->mark = old_mark;
23473f0a83dSXin LI 	      menu->marklen = (short)((old_mark != 0) ? strlen(old_mark) : 0);
2350e3d5408SPeter Wemm 	      RETURN(E_SYSTEM_ERROR);
2360e3d5408SPeter Wemm 	    }
2370e3d5408SPeter Wemm 	}
2380e3d5408SPeter Wemm       else
2390e3d5408SPeter Wemm 	menu->mark = (char *)0;
2400e3d5408SPeter Wemm 
2410e3d5408SPeter Wemm       if ((old_status & _MARK_ALLOCATED) && old_mark)
2420e3d5408SPeter Wemm 	free(old_mark);
2430e3d5408SPeter Wemm 
2440e3d5408SPeter Wemm       if (menu->status & _POSTED)
2450e3d5408SPeter Wemm 	{
2460e3d5408SPeter Wemm 	  _nc_Draw_Menu(menu);
2470e3d5408SPeter Wemm 	  _nc_Show_Menu(menu);
2480e3d5408SPeter Wemm 	}
2490e3d5408SPeter Wemm       else
2500e3d5408SPeter Wemm 	{
2510e3d5408SPeter Wemm 	  /* Recalculate the geometry */
2520e3d5408SPeter Wemm 	  _nc_Calculate_Item_Length_and_Width(menu);
2530e3d5408SPeter Wemm 	}
2540e3d5408SPeter Wemm     }
2550e3d5408SPeter Wemm   else
2560e3d5408SPeter Wemm     {
2574a1a9510SRong-En Fan       returnCode(set_menu_mark(&_nc_Default_Menu, mark));
2580e3d5408SPeter Wemm     }
2590e3d5408SPeter Wemm   RETURN(E_OK);
2600e3d5408SPeter Wemm }
2610e3d5408SPeter Wemm 
2620e3d5408SPeter Wemm /*---------------------------------------------------------------------------
2630e3d5408SPeter Wemm |   Facility      :  libnmenu
2640e3d5408SPeter Wemm |   Function      :  char *menu_mark(const MENU *menu)
2650e3d5408SPeter Wemm |
2660e3d5408SPeter Wemm |   Description   :  Return a pointer to the marker string
2670e3d5408SPeter Wemm |
2680e3d5408SPeter Wemm |   Return Values :  The marker string pointer or NULL if no marker defined
2690e3d5408SPeter Wemm +--------------------------------------------------------------------------*/
2707a656419SBaptiste Daroussin MENU_EXPORT(const char *)
menu_mark(const MENU * menu)2717a69bbfbSPeter Wemm menu_mark(const MENU *menu)
2720e3d5408SPeter Wemm {
27306bfebdeSXin LI   T((T_CALLED("menu_mark(%p)"), (const void *)menu));
2744a1a9510SRong-En Fan   returnPtr(Normalize_Menu(menu)->mark);
2750e3d5408SPeter Wemm }
2760e3d5408SPeter Wemm 
2770e3d5408SPeter Wemm /* m_item_new.c */
278