/* * libcucul Canvas for ultrafast compositing of Unicode letters * Copyright (c) 2002-2006 Sam Hocevar * All Rights Reserved * * $Id$ * * This library is free software; you can redistribute it and/or * modify it under the terms of the Do What The Fuck You Want To * Public License, Version 2, as published by Sam Hocevar. See * http://sam.zoy.org/wtfpl/COPYING for more details. */ /* * This file contains various canvas handling functions such as character * and string drawing. */ #include "config.h" #include "common.h" #if !defined(__KERNEL__) # include /* BUFSIZ */ # include # include # include # if defined(HAVE_ERRNO_H) # include # endif # if defined(HAVE_UNISTD_H) # include # endif # if defined(HAVE_SIGNAL_H) # include # endif # if defined(HAVE_SYS_IOCTL_H) # include # endif #endif #include "cucul.h" #include "cucul_internals.h" /** \brief Print an ASCII or Unicode character. * * This function prints an ASCII or Unicode character at the given * coordinates, using the default foreground and background values. * * If the coordinates are outside the canvas boundaries, nothing is printed. * If the character value is a non-printable character or is outside the * UTF-32 range, it is replaced with a space. To print a sequence of bytes * forming an UTF-8 character instead of an UTF-32 character, use the * cucul_putstr() function instead. * * This function never fails. * * \param cv A handle to the libcucul canvas. * \param x X coordinate. * \param y Y coordinate. * \param ch The character to print. * \return This function always returns 0. */ int cucul_putchar(cucul_canvas_t *cv, int x, int y, unsigned long int ch) { if(x < 0 || x >= (int)cv->width || y < 0 || y >= (int)cv->height) return 0; if((unsigned char)ch < 0x20) ch = 0x20; cv->chars[x + y * cv->width] = ch; cv->attr[x + y * cv->width] = (cv->bgcolor << 16) | cv->fgcolor; return 0; } /** \brief Get the Unicode character at the given coordinates. * * This function gets the ASCII or Unicode value of the character at * the given coordinates. If the value is less or equal to 127 (0x7f), * the character can be printed as ASCII. Otherise, it must be handled * as a UTF-32 value. * * If the coordinates are outside the canvas boundaries, a space (0x20) * is returned. * * This function never fails. * * \param cv A handle to the libcucul canvas. * \param x X coordinate. * \param y Y coordinate. * \param ch The requested character value. * \return The character always returns 0. */ unsigned long int cucul_getchar(cucul_canvas_t *cv, int x, int y) { if(x < 0 || x >= (int)cv->width || y < 0 || y >= (int)cv->height) return 0; return (unsigned long int)cv->chars[x + y * cv->width]; } /** \brief Print a string. * * This function prints an UTF-8 string at the given coordinates, using the * default foreground and background values. The coordinates may be outside * the canvas boundaries (eg. a negative Y coordinate) and the string will * be cropped accordingly if it is too long. * * This function never fails. * * \param cv A handle to the libcucul canvas. * \param x X coordinate. * \param y Y coordinate. * \param s The string to print. * \return This function always returns 0. */ int cucul_putstr(cucul_canvas_t *cv, int x, int y, char const *s) { uint32_t *chars, *attr; unsigned int len; if(y < 0 || y >= (int)cv->height || x >= (int)cv->width) return 0; len = _cucul_strlen_utf8(s); if(x < 0) { if(len < (unsigned int)-x) return 0; len -= -x; s = _cucul_skip_utf8(s, -x); x = 0; } chars = cv->chars + x + y * cv->width; attr = cv->attr + x + y * cv->width; if(x + len >= cv->width) len = cv->width - x; while(len) { *chars++ = cucul_utf8_to_utf32(s, NULL); *attr++ = (cv->bgcolor << 16) | cv->fgcolor; s = _cucul_skip_utf8(s, 1); len--; } return 0; } /** \brief Print a formated string. * * This function formats a string at the given coordinates, using the * default foreground and background values. The coordinates may be outside * the canvas boundaries (eg. a negative Y coordinate) and the string will * be cropped accordingly if it is too long. The syntax of the format * string is the same as for the C printf() function. * * This function never fails. * * \param cv A handle to the libcucul canvas. * \param x X coordinate. * \param y Y coordinate. * \param format The format string to print. * \param ... Arguments to the format string. * \return This function always returns 0. */ int cucul_printf(cucul_canvas_t *cv, int x, int y, char const *format, ...) { char tmp[BUFSIZ]; char *buf = tmp; va_list args; if(y < 0 || y >= (int)cv->height || x >= (int)cv->width) return 0; if(cv->width - x + 1 > BUFSIZ) buf = malloc(cv->width - x + 1); va_start(args, format); #if defined(HAVE_VSNPRINTF) vsnprintf(buf, cv->width - x + 1, format, args); #else vsprintf(buf, format, args); #endif buf[cv->width - x] = '\0'; va_end(args); cucul_putstr(cv, x, y, buf); if(buf != tmp) free(buf); return 0; } /** \brief Clear the canvas. * * This function clears the canvas using the current background colour. * * This function never fails. * * \param cv The canvas to clear. * \return This function always returns 0. */ int cucul_clear_canvas(cucul_canvas_t *cv) { uint32_t color = (cv->bgcolor << 16) | cv->fgcolor; unsigned int n; /* We could use SLsmg_cls() etc., but drawing empty lines is much faster */ for(n = cv->width * cv->height; n--; ) { cv->chars[n] = (uint32_t)' '; cv->attr[n] = color; } return 0; } /** \brief Blit a canvas onto another one. * * This function blits a canvas onto another one at the given coordinates. * An optional mask canvas can be used. * * If an error occurs, -1 is returned and \b errno is set accordingly: * - \c EINVAL A mask was specified but the mask size and source canvas * size do not match. * * \param dst The destination canvas. * \param x X coordinate. * \param y Y coordinate. * \param src The source canvas. * \param mask The mask canvas. * \return 0 in case of success, -1 if an error occurred. */ int cucul_blit(cucul_canvas_t *dst, int x, int y, cucul_canvas_t const *src, cucul_canvas_t const *mask) { int i, j, starti, startj, endi, endj; if(mask && (src->width != mask->width || src->height != mask->height)) { #if defined(HAVE_ERRNO_H) errno = EINVAL; #endif return -1; } starti = x < 0 ? -x : 0; startj = y < 0 ? -y : 0; endi = (x + src->width >= dst->width) ? dst->width - x : src->width; endj = (y + src->height >= dst->height) ? dst->height - y : src->height; if((unsigned int)starti > src->width || (unsigned int)startj > src->height || starti >= endi || startj >= endj) return 0; for(j = startj; j < endj; j++) { if(mask) { for(i = starti; i < endi; i++) { if(mask->chars[j * src->width + i] == (uint32_t)' ') continue; dst->chars[(j + y) * dst->width + (i + x)] = src->chars[j * src->width + i]; dst->attr[(j + y) * dst->width + (i + x)] = src->attr[j * src->width + i]; } } else { memcpy(dst->chars + (j + y) * dst->width + starti + x, src->chars + j * src->width + starti, (endi - starti) * 4); memcpy(dst->attr + (j + y) * dst->width + starti + x, src->attr + j * src->width + starti, (endi - starti) * 4); } } return 0; } /** \brief Set a canvas' new boundaries. * * This function sets new boundaries for a canvas. It can be used to crop a * canvas, to expand it or for combinations of both actions. * * If an error occurs, -1 is returned and \b errno is set accordingly: * - \c EBUSY The canvas is in use by a display driver and cannot be resized. * - \c ENOMEM Not enough memory for the requested canvas size. If this * happens, the canvas handle becomes invalid and should not be used. * * \param cv The canvas to crop. * \param x X coordinate of the top-left corner. * \param y Y coordinate of the top-left corner. * \param w The width of the cropped area. * \param h The height of the cropped area. * \return 0 in case of success, -1 if an error occurred. */ int cucul_set_canvas_boundaries(cucul_canvas_t *cv, int x, int y, unsigned int w, unsigned int h) { cucul_canvas_t *new; unsigned int f, saved_f, framecount; if(cv->refcount) { #if defined(HAVE_ERRNO_H) errno = EBUSY; #endif return -1; } new = cucul_create_canvas(w, h); framecount = cucul_get_canvas_frame_count(cv); saved_f = cv->frame; for(f = 0; f < framecount; f++) { if(f) cucul_create_canvas_frame(new, framecount); cucul_set_canvas_frame(cv, f); cucul_set_canvas_frame(new, f); cucul_blit(new, -x, -y, cv, NULL); free(cv->allchars[f]); free(cv->allattr[f]); } memcpy(cv, new, sizeof(cucul_canvas_t)); free(new); cucul_set_canvas_frame(cv, saved_f); return 0; }