2019-03-16 19:27:45 +01:00
|
|
|
/****************************************************************************
|
2021-06-16 09:22:16 +02:00
|
|
|
* apps/examples/pwfb/pwfb_main.c
|
2019-03-16 19:27:45 +01:00
|
|
|
*
|
2021-06-15 09:09:58 +02:00
|
|
|
* Licensed to the Apache Software Foundation (ASF) under one or more
|
|
|
|
* contributor license agreements. See the NOTICE file distributed with
|
|
|
|
* this work for additional information regarding copyright ownership. The
|
|
|
|
* ASF licenses this file to you under the Apache License, Version 2.0 (the
|
|
|
|
* "License"); you may not use this file except in compliance with the
|
|
|
|
* License. You may obtain a copy of the License at
|
2019-03-16 19:27:45 +01:00
|
|
|
*
|
2021-06-15 09:09:58 +02:00
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
2019-03-16 19:27:45 +01:00
|
|
|
*
|
2021-06-15 09:09:58 +02:00
|
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
|
|
* distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
|
|
|
* WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
|
|
|
* License for the specific language governing permissions and limitations
|
|
|
|
* under the License.
|
2019-03-16 19:27:45 +01:00
|
|
|
*
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Included Files
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
#include <nuttx/config.h>
|
|
|
|
|
|
|
|
#include <sys/types.h>
|
|
|
|
#include <sys/boardctl.h>
|
|
|
|
|
|
|
|
#include <stdint.h>
|
|
|
|
#include <stdbool.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <sched.h>
|
|
|
|
#include <pthread.h>
|
|
|
|
#include <fixedmath.h>
|
|
|
|
#include <errno.h>
|
|
|
|
#include <debug.h>
|
|
|
|
|
|
|
|
#include <nuttx/arch.h>
|
|
|
|
#include <nuttx/board.h>
|
|
|
|
|
|
|
|
#include <nuttx/nx/nx.h>
|
|
|
|
#include <nuttx/nx/nxtk.h>
|
|
|
|
#include <nuttx/nx/nxbe.h>
|
|
|
|
#include <nuttx/nx/nxfonts.h>
|
|
|
|
|
2019-04-10 17:24:08 +02:00
|
|
|
#ifdef CONFIG_NX_SWCURSOR
|
|
|
|
# undef CONFIG_NXWIDGETS_BPP
|
|
|
|
# define CONFIG_NXWIDGETS_BPP CONFIG_EXAMPLES_PWFB_BPP
|
|
|
|
# include "cursor-arrow1-30x30.h"
|
|
|
|
#endif
|
|
|
|
|
2019-03-16 19:27:45 +01:00
|
|
|
#include "pwfb_internal.h"
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Private Data
|
|
|
|
****************************************************************************/
|
|
|
|
|
2019-04-10 20:33:31 +02:00
|
|
|
#if CONFIG_EXAMPLES_PWFB_NWINDOWS > 0
|
2019-03-16 19:27:45 +01:00
|
|
|
static const char g_wndomsg1[] = "NuttX is cool!";
|
2019-04-10 20:33:31 +02:00
|
|
|
#endif
|
|
|
|
#if CONFIG_EXAMPLES_PWFB_NWINDOWS > 1
|
2019-03-16 19:27:45 +01:00
|
|
|
static const char g_wndomsg2[] = "NuttX is fun!";
|
2019-04-10 20:33:31 +02:00
|
|
|
#endif
|
|
|
|
#if CONFIG_EXAMPLES_PWFB_NWINDOWS > 2
|
2019-03-16 19:27:45 +01:00
|
|
|
static const char g_wndomsg3[] = "NuttX is groovy!";
|
2019-04-10 20:33:31 +02:00
|
|
|
#endif
|
2019-03-16 19:27:45 +01:00
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Private Functions
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Name: pwfb_server_initialize
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
static bool pwfb_server_initialize(FAR struct pwfb_state_s *st)
|
|
|
|
{
|
|
|
|
struct sched_param param;
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
/* Set the client task priority */
|
|
|
|
|
2019-03-17 20:51:03 +01:00
|
|
|
param.sched_priority = CONFIG_EXAMPLES_PWFB_CLIENT_PRIO;
|
2019-03-16 19:27:45 +01:00
|
|
|
ret = sched_setparam(0, ¶m);
|
|
|
|
if (ret < 0)
|
|
|
|
{
|
|
|
|
printf("pwfb_server_initialize: ERROR: "
|
|
|
|
"sched_setparam failed: %d\n" ,
|
|
|
|
ret);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Start the NX server kernel thread */
|
|
|
|
|
|
|
|
ret = boardctl(BOARDIOC_NX_START, 0);
|
|
|
|
if (ret < 0)
|
|
|
|
{
|
|
|
|
printf("pwfb_server_initialize: ERROR: "
|
|
|
|
"Failed to start the NX server: %d\n",
|
|
|
|
errno);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Connect to the server */
|
|
|
|
|
|
|
|
st->hnx = nx_connect();
|
|
|
|
if (st->hnx)
|
|
|
|
{
|
|
|
|
#ifdef CONFIG_VNCSERVER
|
|
|
|
/* Setup the VNC server to support keyboard/mouse inputs */
|
|
|
|
|
2019-04-02 15:06:06 +02:00
|
|
|
struct boardioc_vncstart_s vnc =
|
|
|
|
{
|
|
|
|
0, st->hnx
|
|
|
|
};
|
|
|
|
|
|
|
|
ret = boardctl(BOARDIOC_VNC_START, (uintptr_t)&vnc);
|
|
|
|
if (ret < 0)
|
|
|
|
{
|
|
|
|
printf("pwfb_server_initialize: ERROR: "
|
|
|
|
"boardctl(BOARDIOC_VNC_START) failed: %d\n",
|
|
|
|
ret);
|
|
|
|
|
|
|
|
nx_disconnect(st->hnx);
|
|
|
|
g_exitcode = NXEXIT_FBINITIALIZE;
|
|
|
|
return ERROR;
|
|
|
|
}
|
2019-03-16 19:27:45 +01:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
printf("pwfb_server_initialize: ERROR: "
|
|
|
|
"nx_connect failed: %d\n",
|
|
|
|
errno);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Name: pwfb_listener_initialize
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
static bool pwfb_listener_initialize(FAR struct pwfb_state_s *st)
|
|
|
|
{
|
|
|
|
struct sched_param param;
|
|
|
|
pthread_attr_t attr;
|
|
|
|
pthread_t thread;
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
/* Start a separate thread to listen for server events. This is probably
|
|
|
|
* the least efficient way to do this, but it makes this example flow more
|
|
|
|
* smoothly.
|
|
|
|
*/
|
|
|
|
|
2020-01-02 13:09:50 +01:00
|
|
|
pthread_attr_init(&attr);
|
2019-03-17 20:51:03 +01:00
|
|
|
param.sched_priority = CONFIG_EXAMPLES_PWFB_LISTENER_PRIO;
|
2020-01-02 13:09:50 +01:00
|
|
|
pthread_attr_setschedparam(&attr, ¶m);
|
|
|
|
pthread_attr_setstacksize(&attr, CONFIG_EXAMPLES_PWFB_LISTENER_STACKSIZE);
|
2019-03-16 19:27:45 +01:00
|
|
|
|
|
|
|
ret = pthread_create(&thread, &attr, pwfb_listener, st);
|
|
|
|
if (ret != 0)
|
|
|
|
{
|
|
|
|
printf("pwfb_listener_initialize: ERROR: "
|
|
|
|
"pthread_create failed: %d\n",
|
|
|
|
ret);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Don't return until we are connected to the server */
|
|
|
|
|
|
|
|
while (!st->connected)
|
|
|
|
{
|
|
|
|
/* Wait for the listener thread to wake us up when we really
|
|
|
|
* are connected.
|
|
|
|
*/
|
|
|
|
|
2020-01-02 13:09:50 +01:00
|
|
|
sem_wait(&st->semevent);
|
2019-03-16 19:27:45 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Name: pwfb_state_initialize
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
static bool pwfb_state_initialize(FAR struct pwfb_state_s *st)
|
|
|
|
{
|
|
|
|
FAR const struct nx_font_s *fontset;
|
|
|
|
|
|
|
|
/* Initialize semaphores */
|
|
|
|
|
|
|
|
sem_init(&st->semevent, 0, 0);
|
|
|
|
|
|
|
|
/* Initialize color information */
|
|
|
|
|
|
|
|
st->wndo[0].color[0] = CONFIG_EXAMPLES_PWFB_COLOR1;
|
|
|
|
st->wndo[1].color[0] = CONFIG_EXAMPLES_PWFB_COLOR2;
|
|
|
|
st->wndo[2].color[0] = CONFIG_EXAMPLES_PWFB_COLOR3;
|
2019-03-18 19:14:49 +01:00
|
|
|
st->color[0] = CONFIG_EXAMPLES_PWFB_TBCOLOR;
|
2019-03-16 19:27:45 +01:00
|
|
|
|
2019-03-20 19:36:46 +01:00
|
|
|
/* Connect each window to the font cache. They cannot share the
|
2020-02-23 05:51:44 +01:00
|
|
|
* font cache because of the differing background colors.
|
2019-03-16 19:27:45 +01:00
|
|
|
*/
|
|
|
|
|
|
|
|
st->wndo[0].fcache = nxf_cache_connect(CONFIG_EXAMPLES_PWFB_FONTID,
|
|
|
|
CONFIG_EXAMPLES_PWFB_FONTCOLOR,
|
|
|
|
CONFIG_EXAMPLES_PWFB_COLOR1,
|
|
|
|
CONFIG_EXAMPLES_PWFB_BPP, 8);
|
|
|
|
if (st->wndo[0].fcache == NULL)
|
|
|
|
{
|
|
|
|
printf("pwfb_state_initialize: ERROR: "
|
|
|
|
"Failed to connect to font cache for window 1,"
|
|
|
|
"font ID %d: %d\n",
|
|
|
|
CONFIG_EXAMPLES_PWFB_FONTID, errno);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
st->wndo[1].fcache = nxf_cache_connect(CONFIG_EXAMPLES_PWFB_FONTID,
|
|
|
|
CONFIG_EXAMPLES_PWFB_FONTCOLOR,
|
|
|
|
CONFIG_EXAMPLES_PWFB_COLOR2,
|
|
|
|
CONFIG_EXAMPLES_PWFB_BPP, 8);
|
|
|
|
if (st->wndo[1].fcache == NULL)
|
|
|
|
{
|
|
|
|
printf("pwfb_state_initialize: ERROR: "
|
|
|
|
"Failed to connect to font cache for window 2,"
|
|
|
|
"font ID %d: %d\n",
|
|
|
|
CONFIG_EXAMPLES_PWFB_FONTID, errno);
|
|
|
|
goto errout_with_fcache1;
|
|
|
|
}
|
|
|
|
|
|
|
|
st->wndo[2].fcache = nxf_cache_connect(CONFIG_EXAMPLES_PWFB_FONTID,
|
|
|
|
CONFIG_EXAMPLES_PWFB_FONTCOLOR,
|
|
|
|
CONFIG_EXAMPLES_PWFB_COLOR3,
|
|
|
|
CONFIG_EXAMPLES_PWFB_BPP, 8);
|
|
|
|
if (st->wndo[2].fcache == NULL)
|
|
|
|
{
|
|
|
|
printf("pwfb_state_initialize: ERROR: "
|
|
|
|
"Failed to connect to font cache for window 3,"
|
|
|
|
"font ID %d: %d\n",
|
|
|
|
CONFIG_EXAMPLES_PWFB_FONTID, errno);
|
|
|
|
goto errout_with_fcache2;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Get the handle of the font managed by the font caches. Since the
|
|
|
|
* font is used, the same font handle can be shared.
|
|
|
|
*/
|
|
|
|
|
|
|
|
st->hfont = nxf_cache_getfonthandle(st->wndo[0].fcache);
|
|
|
|
if (st->hfont == NULL)
|
|
|
|
{
|
|
|
|
printf("pwfb_state_initialize: ERROR: "
|
|
|
|
"Failed to get handle for font ID %d: %d\n",
|
|
|
|
CONFIG_EXAMPLES_PWFB_FONTID, errno);
|
|
|
|
goto errout_with_fcache3;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Get information about the font set being used and save this in the
|
|
|
|
* state structure
|
|
|
|
*/
|
|
|
|
|
|
|
|
fontset = nxf_getfontset(st->hfont);
|
|
|
|
st->fheight = fontset->mxheight;
|
|
|
|
st->fwidth = fontset->mxwidth;
|
|
|
|
st->spwidth = fontset->spwidth;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
|
|
|
|
errout_with_fcache3:
|
|
|
|
nxf_cache_disconnect(st->wndo[2].fcache);
|
|
|
|
|
|
|
|
errout_with_fcache2:
|
|
|
|
nxf_cache_disconnect(st->wndo[1].fcache);
|
|
|
|
|
|
|
|
errout_with_fcache1:
|
|
|
|
nxf_cache_disconnect(st->wndo[0].fcache);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Name: pwfb_putc
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
static bool pwfb_putc(FAR struct pwfb_state_s *st,
|
|
|
|
FAR struct pwfb_window_s *wndo,
|
|
|
|
FAR struct nxgl_point_s *fpos,
|
|
|
|
uint8_t ch)
|
|
|
|
{
|
|
|
|
FAR const struct nxfonts_glyph_s *glyph;
|
|
|
|
FAR const struct nx_fontbitmap_s *fbm;
|
|
|
|
struct nxgl_rect_s bounds;
|
|
|
|
struct nxgl_size_s fsize;
|
|
|
|
FAR const void *src;
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
/* Find (or create) the matching glyph */
|
|
|
|
|
|
|
|
glyph = nxf_cache_getglyph(wndo->fcache, ch);
|
|
|
|
if (glyph == NULL)
|
|
|
|
{
|
|
|
|
/* No, there is no font for this code. Treat it like a space. */
|
|
|
|
|
|
|
|
fpos->x += st->spwidth;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Get information about the glyph? */
|
|
|
|
|
|
|
|
fbm = nxf_getbitmap(st->hfont, ch);
|
|
|
|
if (fbm == NULL)
|
|
|
|
{
|
|
|
|
/* Should not happen ceause we already know that the character has a
|
|
|
|
* glyph.
|
|
|
|
*/
|
|
|
|
|
|
|
|
fpos->x += st->spwidth;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Get the font size */
|
|
|
|
|
|
|
|
fsize.w = fbm->metric.width + fbm->metric.xoffset;
|
|
|
|
fsize.h = fbm->metric.height + fbm->metric.yoffset;
|
|
|
|
|
|
|
|
/* Construct a bounding box for the glyph */
|
|
|
|
|
|
|
|
bounds.pt1.x = fpos->x;
|
|
|
|
bounds.pt1.y = fpos->y;
|
|
|
|
bounds.pt2.x = fpos->x + fsize.w - 1;
|
|
|
|
bounds.pt2.y = fpos->y + fsize.h - 1;
|
|
|
|
|
|
|
|
/* Render the glyph into the window */
|
|
|
|
|
2022-10-08 10:31:21 +02:00
|
|
|
src = glyph->bitmap;
|
2019-03-16 19:27:45 +01:00
|
|
|
ret = nxtk_bitmapwindow(wndo->hwnd, &bounds, &src,
|
|
|
|
fpos, (unsigned int)glyph->stride);
|
|
|
|
if (ret < 0)
|
|
|
|
{
|
|
|
|
printf("pwfb_putc: ERROR: "
|
|
|
|
"nxtk_bitmapwindow failed: %d\n",
|
|
|
|
errno);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Set up the next character position */
|
|
|
|
|
|
|
|
fpos->x += glyph->width;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Name: pwfb_configure_window
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
static bool pwfb_configure_window(FAR struct pwfb_state_s *st, int wndx,
|
|
|
|
FAR struct nxgl_size_s *size,
|
|
|
|
FAR struct nxgl_point_s *pos,
|
|
|
|
FAR const char *text,
|
|
|
|
double deltax, double deltay)
|
|
|
|
{
|
|
|
|
FAR struct pwfb_window_s *wndo = &st->wndo[wndx];
|
|
|
|
FAR const char *ptr;
|
|
|
|
struct nxgl_rect_s rect;
|
|
|
|
struct nxgl_point_s textpos;
|
2019-03-18 19:14:49 +01:00
|
|
|
nxgl_coord_t tbheight;
|
2019-03-16 19:27:45 +01:00
|
|
|
int ret;
|
|
|
|
|
|
|
|
/* Set the size of the window */
|
|
|
|
|
|
|
|
printf("pwfb_configure_window: Set window %d size to (%d,%d)\n",
|
|
|
|
wndx + 1, size->w, size->h);
|
|
|
|
|
|
|
|
ret = nxtk_setsize(wndo->hwnd, size);
|
|
|
|
if (ret < 0)
|
|
|
|
{
|
|
|
|
printf("pwfb_configure_window: ERROR: "
|
2019-03-17 21:39:04 +01:00
|
|
|
"nxtk_setsize failed: %d\n", errno);
|
2019-03-16 19:27:45 +01:00
|
|
|
goto errout_with_hwnd;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Set the position of window */
|
|
|
|
|
|
|
|
printf("pwfb_configure_window: Set window %d position to (%d,%d)\n",
|
|
|
|
wndx + 1, pos->x, pos->y);
|
|
|
|
|
|
|
|
ret = nxtk_setposition(wndo->hwnd, pos);
|
|
|
|
if (ret < 0)
|
|
|
|
{
|
|
|
|
printf("pwfb_configure_window: ERROR: "
|
|
|
|
"nxtk_setposition failed: %d\n",
|
|
|
|
errno);
|
|
|
|
goto errout_with_hwnd;
|
|
|
|
}
|
|
|
|
|
2019-03-18 19:14:49 +01:00
|
|
|
/* Create a toolbar */
|
|
|
|
|
|
|
|
tbheight = size->h >> 3;
|
|
|
|
ret = nxtk_opentoolbar(wndo->hwnd, tbheight, &g_pwfb_tbcb, st);
|
|
|
|
if (ret < 0)
|
|
|
|
{
|
|
|
|
printf("nxeq_opentoolbar: nxtk_opentoolbar failed: %d\n", errno);
|
|
|
|
}
|
2019-03-16 22:54:08 +01:00
|
|
|
|
2019-03-20 19:36:46 +01:00
|
|
|
/* There is a race condition here we resolve by making the main thread
|
|
|
|
* lowest in priority. In order for the size and position to take effect,
|
|
|
|
* a command is sent to server which responds with an event. So we need
|
|
|
|
* to be synchronized at this point or the following fill will fail because
|
|
|
|
* it depends on current knowlede of the size and position.
|
2019-03-25 20:01:52 +01:00
|
|
|
* REVISIT: Use nx_synch()!
|
2019-03-20 19:36:46 +01:00
|
|
|
*/
|
|
|
|
|
2019-03-16 19:27:45 +01:00
|
|
|
/* Create a bounding box. This is actually too large because it does not
|
2019-04-10 20:33:31 +02:00
|
|
|
* account for the border widths. However, NX should clip the fill to
|
2019-03-16 19:27:45 +01:00
|
|
|
* stay within the frame.
|
|
|
|
*/
|
|
|
|
|
|
|
|
rect.pt1.x = 0;
|
|
|
|
rect.pt1.y = 0;
|
|
|
|
rect.pt2.x = size->w - 1;
|
2019-03-18 19:14:49 +01:00
|
|
|
rect.pt2.y = size->h - tbheight - 1;
|
2019-03-16 19:27:45 +01:00
|
|
|
|
|
|
|
/* Fill the window with the selected color */
|
|
|
|
|
|
|
|
ret = nxtk_fillwindow(wndo->hwnd, &rect, wndo->color);
|
|
|
|
if (ret < 0)
|
|
|
|
{
|
|
|
|
printf("pwfb_configure_window: ERROR: "
|
|
|
|
"nxtk_fillwindow failed: %d\n",
|
|
|
|
errno);
|
|
|
|
goto errout_with_hwnd;
|
|
|
|
}
|
|
|
|
|
2019-03-18 19:14:49 +01:00
|
|
|
/* Fill the toolbar with the selected color */
|
|
|
|
|
|
|
|
rect.pt2.y = tbheight - 1;
|
|
|
|
|
|
|
|
ret = nxtk_filltoolbar(wndo->hwnd, &rect, st->color);
|
|
|
|
if (ret < 0)
|
|
|
|
{
|
|
|
|
printf("pwfb_configure_window: ERROR: "
|
|
|
|
"nxtk_filltoobar failed: %d\n",
|
|
|
|
errno);
|
|
|
|
goto errout_with_hwnd;
|
|
|
|
}
|
|
|
|
|
2019-03-16 19:27:45 +01:00
|
|
|
/* Add the text to the display, character at a time */
|
|
|
|
|
|
|
|
textpos.x = st->spwidth;
|
|
|
|
textpos.y = st->fheight;
|
|
|
|
|
|
|
|
for (ptr = text; *ptr != '\0'; ptr++)
|
|
|
|
{
|
|
|
|
if (!pwfb_putc(st, wndo, &textpos, (uint8_t)*ptr))
|
|
|
|
{
|
|
|
|
printf("pwfb_configure_window: ERROR: "
|
|
|
|
"pwfb_putc failed\n");
|
|
|
|
goto errout_with_hwnd;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-04-10 20:33:31 +02:00
|
|
|
/* Set up for motion.
|
|
|
|
* REVISIT: The vertical limits, xmax andymax, seems to be off
|
|
|
|
* by about the height of the toolbar.
|
|
|
|
*/
|
2019-03-16 19:27:45 +01:00
|
|
|
|
2019-03-16 21:14:07 +01:00
|
|
|
wndo->xmax = itob16(st->xres - size->w - 1);
|
|
|
|
wndo->ymax = itob16(st->yres - size->h - 1);
|
|
|
|
wndo->ypos = itob16(pos->y);
|
|
|
|
wndo->xpos = itob16(pos->x);
|
|
|
|
wndo->deltax = dtob16(deltax);
|
|
|
|
wndo->deltay = dtob16(deltay);
|
2019-03-16 19:27:45 +01:00
|
|
|
|
|
|
|
return true;
|
|
|
|
|
|
|
|
errout_with_hwnd:
|
|
|
|
printf("pwfb_configure_window: Close window %d\n", wndx + 1);
|
|
|
|
|
|
|
|
ret = nxtk_closewindow(wndo->hwnd);
|
|
|
|
if (ret < 0)
|
|
|
|
{
|
|
|
|
printf("pwfb_configure_window: ERROR: "
|
|
|
|
"nxtk_closewindow failed: %d\n",
|
|
|
|
errno);
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2019-04-10 17:24:08 +02:00
|
|
|
/****************************************************************************
|
|
|
|
* Name: pwfb_configure_cursor
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
#ifdef CONFIG_NX_SWCURSOR
|
|
|
|
static bool pwfb_configure_cursor(FAR struct pwfb_state_s *st,
|
|
|
|
FAR struct nxgl_point_s *pos,
|
|
|
|
double deltax, double deltay)
|
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
/* Initialize the data structure */
|
|
|
|
|
|
|
|
st->cursor.state = PFWB_CURSOR_MOVING;
|
|
|
|
st->cursor.countdown = CURSOR_MOVING_DELAY;
|
|
|
|
st->cursor.blinktime = 0;
|
2019-04-10 23:00:36 +02:00
|
|
|
st->cursor.xmax = itob16(st->xres - g_arrow1Cursor.size.w - 1);
|
|
|
|
st->cursor.ymax = itob16(st->yres - g_arrow1Cursor.size.h - 1);
|
|
|
|
st->cursor.xpos = itob16(pos->x);
|
|
|
|
st->cursor.ypos = itob16(pos->y);
|
|
|
|
st->cursor.deltax = dtob16(deltax);
|
|
|
|
st->cursor.deltay = dtob16(deltay);
|
2019-04-10 17:24:08 +02:00
|
|
|
|
|
|
|
/* Set the cursor image */
|
|
|
|
|
|
|
|
ret = nxcursor_setimage(st->hnx, &g_arrow1Cursor);
|
|
|
|
if (ret < 0)
|
|
|
|
{
|
|
|
|
printf("pwfb_configure_cursor: ERROR: "
|
|
|
|
"nxcursor_setimage failed: %d\n",
|
|
|
|
errno);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Set the cursor position */
|
|
|
|
|
|
|
|
ret = nxcursor_setposition(st->hnx, pos);
|
|
|
|
if (ret < 0)
|
|
|
|
{
|
|
|
|
printf("pwfb_configure_cursor: ERROR: "
|
|
|
|
"nxcursor_setposition failed: %d\n",
|
|
|
|
errno);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Enable the cursor */
|
|
|
|
|
2019-04-10 23:00:36 +02:00
|
|
|
ret = nxcursor_enable(st->hnx, true);
|
2019-04-10 17:24:08 +02:00
|
|
|
if (ret < 0)
|
|
|
|
{
|
|
|
|
printf("pwfb_configure_cursor: ERROR: "
|
|
|
|
"nxcursor_enable failed: %d\n",
|
|
|
|
errno);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2019-03-16 19:27:45 +01:00
|
|
|
/****************************************************************************
|
|
|
|
* Public Functions
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Name: pwfb_main
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
int main(int argc, FAR char *argv[])
|
|
|
|
{
|
|
|
|
struct pwfb_state_s wstate;
|
|
|
|
struct nxgl_size_s size;
|
|
|
|
struct nxgl_point_s pos;
|
|
|
|
nxgl_mxpixel_t color;
|
|
|
|
int errcode = EXIT_SUCCESS;
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
/* Connect to the NX server */
|
|
|
|
|
|
|
|
memset(&wstate, 0, sizeof(struct pwfb_state_s));
|
|
|
|
if (!pwfb_server_initialize(&wstate) || wstate.hnx == NULL)
|
|
|
|
{
|
|
|
|
printf("pwfb_main: ERROR: "
|
|
|
|
"Failed to get NX handle\n");
|
|
|
|
goto errout;
|
|
|
|
}
|
|
|
|
|
|
|
|
printf("pwfb_main: NX handle=%p\n", wstate.hnx);
|
|
|
|
|
|
|
|
/* Start the listener thread */
|
|
|
|
|
|
|
|
if (!pwfb_listener_initialize(&wstate))
|
|
|
|
{
|
|
|
|
printf("pwfb_main: ERROR: "
|
|
|
|
"pwfb_listener_initialize failed\n");
|
|
|
|
goto errout_with_nx;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Initialize the window state, colors, font cache, etc. */
|
|
|
|
|
|
|
|
if (!pwfb_state_initialize(&wstate))
|
|
|
|
{
|
|
|
|
printf("pwfb_main: ERROR: "
|
|
|
|
"pwfb_state_initialize failed\n");
|
|
|
|
goto errout_with_nx;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Set the background to the configured background color */
|
|
|
|
|
|
|
|
printf("pwfb_main: Set background color=%d\n",
|
|
|
|
CONFIG_EXAMPLES_PWFB_BGCOLOR);
|
|
|
|
|
|
|
|
color = CONFIG_EXAMPLES_PWFB_BGCOLOR;
|
|
|
|
ret = nx_setbgcolor(wstate.hnx, &color);
|
|
|
|
if (ret < 0)
|
|
|
|
{
|
|
|
|
printf("pwfb_main: nx_setbgcolor failed: %d\n", errno);
|
|
|
|
goto errout_with_fontcache;
|
|
|
|
}
|
|
|
|
|
2019-04-10 20:33:31 +02:00
|
|
|
#if CONFIG_EXAMPLES_PWFB_NWINDOWS > 0
|
2019-03-16 19:27:45 +01:00
|
|
|
/* Open window 1 */
|
|
|
|
|
|
|
|
printf("pwfb_main: Open window 1\n");
|
|
|
|
|
|
|
|
wstate.wndo[0].hwnd = nxtk_openwindow(wstate.hnx, NXBE_WINDOW_RAMBACKED,
|
2019-03-18 19:14:49 +01:00
|
|
|
&g_pwfb_wncb, (FAR void *)&wstate);
|
2019-03-16 19:27:45 +01:00
|
|
|
if (wstate.wndo[0].hwnd == NULL)
|
|
|
|
{
|
|
|
|
printf("pwfb_main: ERROR: "
|
|
|
|
"nxtk_openwindow failed: %d\n",
|
|
|
|
errno);
|
|
|
|
goto errout_with_fontcache;
|
|
|
|
}
|
|
|
|
|
|
|
|
printf("pwfb_main: hwnd1=%p\n", wstate.wndo[0].hwnd);
|
|
|
|
|
|
|
|
/* Wait until we receive the screen resolution from the server. We only
|
|
|
|
* need to do this once after opening the first window.
|
|
|
|
*/
|
|
|
|
|
|
|
|
while (!wstate.haveres)
|
|
|
|
{
|
2020-01-02 13:09:50 +01:00
|
|
|
sem_wait(&wstate.semevent);
|
2019-03-16 19:27:45 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
printf("pwfb_main: Screen resolution (%d,%d)\n",
|
|
|
|
wstate.xres, wstate.yres);
|
|
|
|
|
|
|
|
/* Configure window 1 */
|
|
|
|
|
|
|
|
size.w = wstate.xres / 2;
|
|
|
|
size.h = wstate.yres / 2;
|
|
|
|
|
2019-03-16 21:14:07 +01:00
|
|
|
pos.x = wstate.xres / 8;
|
|
|
|
pos.y = wstate.yres / 8;
|
2019-03-16 19:27:45 +01:00
|
|
|
|
2019-03-16 21:14:07 +01:00
|
|
|
if (!pwfb_configure_window(&wstate, 0, &size, &pos, g_wndomsg1, 4.200, 4.285))
|
2019-03-16 19:27:45 +01:00
|
|
|
{
|
|
|
|
printf("pwfb_main: ERROR: "
|
|
|
|
"pwfb_configure_window failed for window 1\n");
|
2019-03-20 19:36:46 +01:00
|
|
|
goto errout_with_hwnd1;
|
2019-03-16 19:27:45 +01:00
|
|
|
}
|
2019-04-10 20:33:31 +02:00
|
|
|
#endif
|
2019-03-16 19:27:45 +01:00
|
|
|
|
2019-04-10 20:33:31 +02:00
|
|
|
#if CONFIG_EXAMPLES_PWFB_NWINDOWS > 1
|
2019-03-16 19:27:45 +01:00
|
|
|
/* Open window 2 */
|
|
|
|
|
|
|
|
printf("pwfb_main: Open window 2\n");
|
|
|
|
|
|
|
|
wstate.wndo[1].hwnd = nxtk_openwindow(wstate.hnx, NXBE_WINDOW_RAMBACKED,
|
2019-03-18 19:14:49 +01:00
|
|
|
&g_pwfb_wncb, (FAR void *)&wstate);
|
2019-03-16 19:27:45 +01:00
|
|
|
if (wstate.wndo[1].hwnd == NULL)
|
|
|
|
{
|
|
|
|
printf("pwfb_main: ERROR: "
|
|
|
|
"nxtk_openwindow failed: %d\n",
|
|
|
|
errno);
|
|
|
|
goto errout_with_hwnd1;
|
|
|
|
}
|
|
|
|
|
|
|
|
printf("pwfb_main: hwnd1=%p\n", wstate.wndo[1].hwnd);
|
|
|
|
|
|
|
|
/* Configure window 2 (same size) */
|
|
|
|
|
|
|
|
pos.x = wstate.xres / 4;
|
|
|
|
pos.y = wstate.yres / 4;
|
|
|
|
|
2019-03-16 21:14:07 +01:00
|
|
|
if (!pwfb_configure_window(&wstate, 1, &size, &pos, g_wndomsg2, -3.317, 5.0))
|
2019-03-16 19:27:45 +01:00
|
|
|
{
|
|
|
|
printf("pwfb_main: ERROR: "
|
|
|
|
"pwfb_configure_window failed for window 2\n");
|
|
|
|
goto errout_with_hwnd2;
|
|
|
|
}
|
2019-04-10 20:33:31 +02:00
|
|
|
#endif
|
2019-03-16 19:27:45 +01:00
|
|
|
|
2019-04-10 20:33:31 +02:00
|
|
|
#if CONFIG_EXAMPLES_PWFB_NWINDOWS > 2
|
2019-03-16 19:27:45 +01:00
|
|
|
/* Open window 3 */
|
|
|
|
|
|
|
|
printf("pwfb_main: Open window 3\n");
|
|
|
|
|
|
|
|
wstate.wndo[2].hwnd = nxtk_openwindow(wstate.hnx, NXBE_WINDOW_RAMBACKED,
|
2019-03-18 19:14:49 +01:00
|
|
|
&g_pwfb_wncb, (FAR void *)&wstate);
|
2019-03-16 19:27:45 +01:00
|
|
|
if (wstate.wndo[2].hwnd == NULL)
|
|
|
|
{
|
|
|
|
printf("pwfb_main: ERROR: "
|
|
|
|
"nxtk_openwindow failed: %d\n",
|
|
|
|
errno);
|
|
|
|
goto errout_with_hwnd2;
|
|
|
|
}
|
|
|
|
|
|
|
|
printf("pwfb_main: hwnd2=%p\n", wstate.wndo[2].hwnd);
|
|
|
|
|
|
|
|
/* Configure window 3 (same size) */
|
|
|
|
|
|
|
|
pos.x = (3 * wstate.xres) / 8;
|
|
|
|
pos.y = (3 * wstate.yres) / 8;
|
|
|
|
|
2019-03-16 21:14:07 +01:00
|
|
|
if (!pwfb_configure_window(&wstate, 2, &size, &pos, g_wndomsg3, 4.600, -3.852))
|
2019-03-16 19:27:45 +01:00
|
|
|
{
|
|
|
|
printf("pwfb_main: ERROR: "
|
|
|
|
"pwfb_configure_window failed for window 2\n");
|
|
|
|
goto errout_with_hwnd3;
|
|
|
|
}
|
2019-04-10 20:33:31 +02:00
|
|
|
#endif
|
2019-03-16 19:27:45 +01:00
|
|
|
|
2019-04-10 17:24:08 +02:00
|
|
|
#ifdef CONFIG_NX_SWCURSOR
|
|
|
|
/* Configure the software cursor */
|
|
|
|
|
|
|
|
pos.x = wstate.xres / 2;
|
2019-04-10 23:00:36 +02:00
|
|
|
pos.y = wstate.yres / 2;
|
2019-04-10 17:24:08 +02:00
|
|
|
|
|
|
|
if (!pwfb_configure_cursor(&wstate, &pos, 2.900, -5.253))
|
|
|
|
{
|
|
|
|
printf("pwfb_main: ERROR: "
|
|
|
|
"pwfb_configure_cursor failed for window 2\n");
|
2019-04-10 20:33:31 +02:00
|
|
|
goto errout_with_hwnds;
|
2019-04-10 17:24:08 +02:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2019-03-16 19:27:45 +01:00
|
|
|
/* Now loop animating the windows */
|
|
|
|
|
|
|
|
for (; ; )
|
|
|
|
{
|
2019-03-17 20:51:03 +01:00
|
|
|
usleep(CONFIG_EXAMPLES_PWFB_RATECONTROL * 1000);
|
2019-03-16 19:27:45 +01:00
|
|
|
if (!pwfb_motion(&wstate))
|
|
|
|
{
|
|
|
|
printf("pwfb_main: ERROR:"
|
|
|
|
"pwfb_motion failed\n");
|
2019-04-10 17:24:08 +02:00
|
|
|
goto errout_with_cursor;
|
2019-03-16 19:27:45 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
errcode = EXIT_SUCCESS;
|
|
|
|
|
2019-04-10 17:24:08 +02:00
|
|
|
errout_with_cursor:
|
|
|
|
#ifdef CONFIG_NX_SWCURSOR
|
|
|
|
/* Disable the cursor */
|
|
|
|
|
2020-01-02 13:09:50 +01:00
|
|
|
nxcursor_enable(wstate.hnx, false);
|
2019-04-10 20:33:31 +02:00
|
|
|
|
|
|
|
errout_with_hwnds:
|
2019-04-10 17:24:08 +02:00
|
|
|
#endif
|
|
|
|
|
2019-04-10 20:33:31 +02:00
|
|
|
#if CONFIG_EXAMPLES_PWFB_NWINDOWS > 2
|
2019-03-16 19:27:45 +01:00
|
|
|
/* Close window 3 */
|
|
|
|
|
|
|
|
errout_with_hwnd3:
|
|
|
|
printf("pwfb_main: Close window #2\n");
|
|
|
|
|
|
|
|
ret = nxtk_closewindow(wstate.wndo[2].hwnd);
|
|
|
|
if (ret < 0)
|
|
|
|
{
|
|
|
|
printf("pwfb_main: ERROR: nxtk_closewindow failed: %d\n", errno);
|
|
|
|
}
|
2019-04-10 20:33:31 +02:00
|
|
|
#endif
|
2019-03-16 19:27:45 +01:00
|
|
|
|
2019-04-10 20:33:31 +02:00
|
|
|
#if CONFIG_EXAMPLES_PWFB_NWINDOWS > 1
|
2019-03-16 19:27:45 +01:00
|
|
|
/* Close window 2 */
|
|
|
|
|
|
|
|
errout_with_hwnd2:
|
|
|
|
printf("pwfb_main: Close window #2\n");
|
|
|
|
|
|
|
|
ret = nxtk_closewindow(wstate.wndo[1].hwnd);
|
|
|
|
if (ret < 0)
|
|
|
|
{
|
|
|
|
printf("pwfb_main: ERROR: nxtk_closewindow failed: %d\n", errno);
|
|
|
|
}
|
2019-04-10 20:33:31 +02:00
|
|
|
#endif
|
2019-03-16 19:27:45 +01:00
|
|
|
|
|
|
|
/* Close window1 */
|
|
|
|
|
2019-04-10 20:33:31 +02:00
|
|
|
#if CONFIG_EXAMPLES_PWFB_NWINDOWS > 0
|
2019-03-16 19:27:45 +01:00
|
|
|
errout_with_hwnd1:
|
|
|
|
printf("pwfb_main: Close window #1\n");
|
|
|
|
|
|
|
|
ret = nxtk_closewindow(wstate.wndo[0].hwnd);
|
|
|
|
if (ret < 0)
|
|
|
|
{
|
|
|
|
printf("pwfb_main: ERROR: nxtk_closewindow failed: %d\n", errno);
|
|
|
|
}
|
2019-04-10 20:33:31 +02:00
|
|
|
#endif
|
2019-03-16 19:27:45 +01:00
|
|
|
|
|
|
|
errout_with_fontcache:
|
|
|
|
/* Release the font cache */
|
|
|
|
|
|
|
|
nxf_cache_disconnect(wstate.wndo[0].fcache);
|
|
|
|
nxf_cache_disconnect(wstate.wndo[1].fcache);
|
|
|
|
nxf_cache_disconnect(wstate.wndo[2].fcache);
|
|
|
|
|
|
|
|
errout_with_nx:
|
|
|
|
/* Disconnect from the server */
|
|
|
|
|
|
|
|
printf("pwfb_main: Disconnect from the server\n");
|
|
|
|
nx_disconnect(wstate.hnx);
|
|
|
|
|
|
|
|
errout:
|
|
|
|
return errcode;
|
|
|
|
}
|