44 KiB
Quick Start
helloworld Program of mGNCS
mGNCS packages the main window of MiniGUI and brings it into the control system, therefore, the whole programming will become very simple and easy to learn. Please look at the helloworld program source codes based on mGNCS:
List 1 helloworld.c
/*
** helloworld.c: Sample program for mGNCS Programming Guide
** The first mGNCS application.
**
** Copyright (C) 2009 ~ 2019 FMSoft Technologies.
*/
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
// START_OF_INCS
#include <minigui/common.h>
#include <minigui/minigui.h>
#include <minigui/gdi.h>
#include <minigui/window.h>
#include <mgncs/mgncs.h>
// END_OF_INCS
// START_OF_HANDLERS
static void mymain_onPaint (mWidget *_this, HDC hdc, const CLIPRGN* inv)
{
RECT rt;
GetClientRect (_this->hwnd, &rt);
DrawText (hdc, "Hello, world!", -1, &rt, DT_SINGLELINE|DT_CENTER|DT_VCENTER);
}
static BOOL mymain_onClose (mWidget* _this, int message)
{
DestroyMainWindow (_this->hwnd);
return TRUE;
}
static NCS_EVENT_HANDLER mymain_handlers [] = {
{MSG_PAINT, mymain_onPaint},
{MSG_CLOSE, mymain_onClose},
{0, NULL}
};
// END_OF_HANDLERS
int MiniGUIMain (int argc, const char* argv[])
{
MSG Msg;
ncsInitialize ();
mWidget* mymain = ncsCreateMainWindow (
NCSCTRL_MAINWND, "Hello, world!",
WS_CAPTION | WS_BORDER | WS_VISIBLE,
WS_EX_NONE,
1,
0, 0, 300,200,
HWND_DESKTOP,
0, 0,
NULL,
NULL,
mymain_handlers,
0);
// START_OF_MSGLOOP
while (GetMessage (&Msg, mymain->hwnd)) {
TranslateMessage (&Msg);
DispatchMessage (&Msg);
}
// END_OF_MSGLOOP
MainWindowThreadCleanup (mymain->hwnd);
ncsUninitialize ();
return 0;
}
The program creates a 300x200 main window on the screen and displays “Hello, world!”:
Figure 1 Output of helloworld program
Head File
Besides the four head files of MiniGUI (<minigui/common.h>
,
<minigui/minigui.h>
, <minigui/gdi.h>
and <minigui/window.h>
), helloworld.c
shall contain its specific file:
<mgncs/mgncs.h>
: mGNCS head file
Therefore, normally, head file of a mGNCS program is:
#include <minigui/common.h>
#include <minigui/minigui.h>
#include <minigui/gdi.h>
#include <minigui/window.h>
#include <mgncs/mgncs.h>
Create the Main Window
Same as common MiniGUI program, the entrance point of mGNCS program is also
MiniGUIMain?
. However, before using mGNCS, it is a must to call
ncsInitialize
to register the control of NCS to MiniGUI:
ncsInitialize ();
Afterwards, a main window can be directly created through calling
ncsCreateMainWindow
function, which has many parameters, but a lot of default
parameters can be used for creating default window:
mWidget* mymain = ncsCreateMainWindow (
NCSCTRL_MAINWND, // Class name of the main window, class name macro of NCS control starts with NCSCTRL_
"Hello World!", // Title text of the appointed main window
WS_CAPTION | WS_BORDER | WS_VISIBLE, // Style of the appointed window, create a visible window with title bar and border
WS_EX_NONE, // Extension style of appointed main window, use 0 or WS_EX_NONE to show that there is no extension style
1, // Identifier value of the appointed main window, any integer value >0
0, 0, 300, 200, // Location and size of the appointed main window, x, y, width and height by sequence
HWND_DESKTOP, // Host of the appointed main window, which usually is HWND_DESKTOP
0, 0, // Icon and menu bar sentence handle of the main window, empty by default
NULL, // Initial property information of the appointed main window, empty by default
NULL, // Initial renderer information of the appointed main window, empty by default
mymain_handlers, // Event handler of the appointed main window, see the next section
0); // Additional data of the appointed main window, 0 here
The most important parameter in the text above is event handler
ncsCreateMainWindow
function returns a window object pointer (mWidget*), and
the object is set as pointer of the control basic class for the convenience of
the users’ use.
mWidget
structure provides a hwnd data member, which is window sentence
handle associated with the window object. Users can directly use the sentence
handle to call corresponding MiniGUI API
.
Event Handler
To make the window operates, it is a must to respond to the message of the window. NCS provides a mapping mechanism, which associates an event (message or notification code) with a callback function. Formats of callback functions of different events are different, and NCS has defined interfaces for each event in advance for the convenience of use.
Multiple mappings can be placed in a NCS_EVENT_HANDLER
array (the array ends
with {0,NULL}), and are transferred to the control when window is created.
In the example, we mainly pay attention to two message events:
MSG_PAINT
MSG_CLOSE
Prototypes of the event handlers that the two messages correspond to are:
/**
* \typedef void (*NCS_CB_ONPAINT)(mWidget*, HDC hdc, const PCLIPRGN clip_rgn);
* \brief the callback of event MSG_PAINT
*
* \param mWidget * sender pointer
* \param hdc the DC for painting
* \param clip_rgn invlidate region
*
*/
typedef void (*NCS_CB_ONPAINT)(mWidget*, HDC hdc, const PCLIPRGN clip_rgn);
/**
* \typedef BOOL (*NCS_CB_ONCLOSE)(mWidget*);
* \brief the callback of event MSG_CLOSE
*
* \param mWidget* the sender pointer
*
* \return TRUE - about the message process, FALSE - continue default processing
*/
typedef BOOL (*NCS_CB_ONCLOSE)(mWidget*);
Therefore, we define the two functions, and establish mapping array of the event and event handler:
static void mymain_onPaint (mWidget *_this, HDC hdc, const CLIPRGN* inv)
{
RECT rt;
GetClientRect (_this->hwnd, &rt);
DrawText (hdc, "Hello, world!", -1, &rt, DT_SINGLELINE|DT_CENTER|DT_VCENTER);
}
static BOOL mymain_onClose (mWidget* _this, int message)
{
DestroyMainWindow (_this->hwnd);
return TRUE;
}
static NCS_EVENT_HANDLER mymain_handlers [] = {
{MSG_PAINT, mymain_onPaint},
{MSG_CLOSE, mymain_onClose},
{0, NULL}
};
We can see that compared to MiniGUI common program, mGNCS avoids the use of
multi-layer switch-case nested statement, making it unnecessary for developers
to memorize the meanings of complicated WPARAM
and LPARAM
and meaning of
returned value of event handling process.
If miniStudio is used, the above codes can be generated automatically. What the user needs to do is to implement the corresponding event handler (callback function).
Message Circulation
Finally, to make the program runs, a message circulation is needed, same as for common MiniGUI programming:
while (GetMessage (&Msg, mymain->hwnd)) {
TranslateMessage (&Msg);
DispatchMessage (&Msg);
}
Create Modal Main Window
We know that the main window is divided into modal and non-modal. The above
example program establishes a standard non-modal main window. If we hope to
create modal main window, we only need to use doModal
to replace the existing
message circulation, as shown below:
int MiniGUIMain (int argc, const char* argv[])
{
ncsInitialize ();
mMainWnd* mymain = (mMainWnd*) ncsCreateMainWindow (
NCSCTRL_MAINWND, "Hello, world!",
WS_CAPTION | WS_BORDER | WS_VISIBLE,
WS_EX_NONE,
1,
0, 0, 300, 200,
HWND_DESKTOP,
0, 0,
NULL,
NULL,
mymain_handlers,
0);
_c(mymain)->doModal (mymain, TRUE);
ncsUninitialize ();
return 0;
}
In the above codes
- The effect of _c macro is to take out the class structure (mMainWndClass)
pointer stored by mymain. Because
doModal
is a virtual function ofmMainWnd
class, it is stored in virtual function table (case ofmMainWndClass
). - The first parameter of
doModal
method ismMainWnd*
self, similar to this pointer in C++. The second parameterBOOLautoDestroy
tells ifdoModal
method automatically deletes dialog box object. When the parameter isTRUE
, after callingdoModal
, mymain will be released and cannot be used.
Further Use Template to Create Main Window and Control
We know that MiniGUI provides the function of dialog box template, using
template, control and main window can be defined in the form of data structure,
thus helping with simplifying the compilation of application programs. Based on
MiniGUI, NCS defines window template structure containing more messages, which
is more complicated compared to DLGTEMPLATE
of MiniGUI, but is more
convenient for programming. It needs to be noted that the window template of
NCS is normally automatically generated by miniStudio according to UI designed
by the user. More complicated window template structure will help NCS better
load window from the resource files generated by miniStudio. Although template
is not aiming at developers, we need to know the definition and usage of window
template.
Codes in list p2c1-3 demonstrate how to use template to create a dialog box as shown in the figure below, and demonstrate how to use property and renderer.
Figure 2 Dialog Box Created Using Window Template
List 3 wnd_template.c
/*
** wnd_template.c: Sample program for mGNCS Programming Guide
** Using window template.
**
** Copyright (C) 2009 ~ 2019 FMSoft Technologies.
*/
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <minigui/common.h>
#include <minigui/minigui.h>
#include <minigui/gdi.h>
#include <minigui/window.h>
#include <minigui/control.h>
#include <mgncs/mgncs.h>
#define ID_BTN 101
#define ID_PROG 200
// START_OF_HANDLERS
static BOOL mymain_onCreate (mWidget* _this, DWORD add_data)
{
SetTimer (_this->hwnd, 100, 20);
return TRUE;
}
// START_OF_ONTIMER
static void mymain_onTimer (mWidget *_this, int id, DWORD count)
{
static int pb_pos = 0;
mProgressBar *pb = (mProgressBar*)ncsGetChildObj (_this->hwnd, ID_PROG);
if (pb)
_c(pb)->setProperty(pb, NCSP_PROG_CURPOS, pb_pos++);
}
// END_OF_ONTIMER
static void mymain_onClose (mWidget* _this, int message)
{
DestroyMainWindow (_this->hwnd);
PostQuitMessage (_this->hwnd);
}
static NCS_EVENT_HANDLER mymain_handlers[] = {
{MSG_CREATE, mymain_onCreate},
{MSG_CLOSE, mymain_onClose},
{MSG_TIMER, mymain_onTimer},
{0, NULL}
};
// END_OF_HANDLERS
// START_OF_RDRINFO
static NCS_RDR_ELEMENT btn_rdr_elements[] =
{
{ NCS_MODE_USEFLAT, 1},
{ -1, 0 }
};
static NCS_RDR_INFO btn1_rdr_info[] =
{
{"fashion","fashion", btn_rdr_elements}
};
static NCS_RDR_INFO btn2_rdr_info[] =
{
{"flat", "flat", NULL}
};
// END_OF_RDRINFO
// START_OF_PROPERTIES
static NCS_PROP_ENTRY progress_props [] = {
{NCSP_PROG_MAXPOS, 100},
{NCSP_PROG_MINPOS, 0 },
{NCSP_PROG_LINESTEP, 1},
{NCSP_PROG_CURPOS, 0 },
{ 0, 0 }
};
// END_OF_PROPERTIES
// START_OF_TEMPLATE
static NCS_WND_TEMPLATE _ctrl_templ[] = {
{
NCSCTRL_PROGRESSBAR,
ID_PROG,
10, 10, 290, 30,
WS_BORDER | WS_VISIBLE,
WS_EX_NONE,
"",
progress_props,
NULL,
NULL, NULL, 0, 0
},
{
NCSCTRL_BUTTON,
ID_BTN,
10, 50, 100, 25,
WS_BORDER | WS_VISIBLE,
WS_EX_NONE,
"button1",
NULL,
btn1_rdr_info,
NULL, NULL, 0, 0
},
{
NCSCTRL_BUTTON,
ID_BTN,
200, 50, 100, 25,
WS_VISIBLE,
WS_EX_NONE,
"button2",
NULL,
btn2_rdr_info,
NULL, NULL, 0, 0
},
};
static NCS_MNWND_TEMPLATE mymain_templ = {
NCSCTRL_DIALOGBOX,
1,
0, 0, 320, 110,
WS_CAPTION | WS_BORDER | WS_VISIBLE,
WS_EX_NONE,
"Window Template",
NULL,
NULL,
mymain_handlers,
_ctrl_templ,
sizeof(_ctrl_templ)/sizeof(NCS_WND_TEMPLATE),
0,
0, 0,
};
// END_OF_TEMPLATE
int MiniGUIMain (int argc, const char* argv[])
{
ncsInitialize ();
mDialogBox* mydlg = (mDialogBox *)ncsCreateMainWindowIndirect
(&mymain_templ, HWND_DESKTOP);
_c(mydlg)->doModal (mydlg, TRUE);
ncsUninitialize ();
return 0;
}
The program defines a dialog box containing three controls:
- For the progress bar above, its maximum value and minimum value are set in initialization, and its current location will be refreshed periodically;
- The left button uses fashion renderer, and draws with default property;
- The right button uses flat renderer, and draws with default property.
Template
Function ncsCreateMainWindowIndirect
creates corresponding main window
according to the template. In the above program, the template is
mymain_templ.
The type of mymain_templ
is NCS_MAINWND_TEMPLATE
, and a
pointer points to the template structure array (NCS_WND_TEMPLAT*
) of the
child control. The difference between NCS_WND_TEMPLATE
and
NCS_MAINWND_TEMPLATE
is that the former does not have the data member
specially used for the main window such as HICON
and HMENU
.
Declaration of mymain_templ
is as follows:
static NCS_MAINWND_TEMPLATE mymain_templ = {
NCSCTRL_DIALOGBOX, // Window class name, NCSCTRL_DIALOGBOX is taken here
1, // Window identifier
0, 0, 320, 110, // Window location and size
WS_CAPTION | WS_BORDER | WS_VISIBLE, // Window style
WS_EX_NONE, // Extension style of the window
"Window Template", // Window title
NULL, // Initialization property
NULL, // Renderer information
mymain_handlers, // Window event handler
_ctrl_templ, // Template structure array pointer of the child window
sizeof(_ctrl_templ)/sizeof(NCS_WND_TEMPLATE), // Number of the child window
0, 0, 0 // Additional data of the window, icon sentence handle and menu bar sentence handle
};
It needs to be noted that NCS_WND_TEMPLATE
also contains a pointer member
pointing to child control template structure array, thus multi-level nested
template can be formed.
Set Window Property
In the example, we define the initial property values of the progress bar
control through NCS_PROP_ENTRY
structure array, which include:
- Maximum value of the progress bar
- Minimum value of the progress bar.
- Step value of the progress bar
- Initial location of the progress bar.
Concrete codes are as below:
static NCS_PROP_ENTRY progress_props [] = {
{NCSP_PROG_MAXPOS, 100},
{NCSP_PROG_MINPOS, 0 },
{NCSP_PROG_LINESTEP, 1},
{NCSP_PROG_CURPOS, 0 },
{ 0, 0 }
};
Pay attention to the last member {0, 0} in the above structure array, and NCS shows the end of property information through defining an all null structure.
When defining control, just endow the corresponding window template property member with pointer of the above property structure array, as shown below:
static NCS_WND_TEMPLATE _ctrl_templ[] = {
{
NCSCTRL_PROGRESSBAR,
ID_PROG,
10, 10, 290, 30,
WS_BORDER | WS_VISIBLE,
WS_EX_NONE,
"",
progress_props,
NULL,
NULL, NULL, 0, 0
},
...
};
If MiniGUI intrinsic control set is used, to implement this function, it is a
must to call SendMessage
? in MSG_INITDIALOG
message of the dialog box.
While in the new control set, it is only necessary to set a NCS_PROP_ENTRY
array, and the function can be implemented automatically. It is not only for
reducing code amount and simplifying programming, more importantly, it provides
a uniform interface. Users can store the initial information into the external
files in a unified way through this interface. Resources management module of
NCS can finish similar functions, and miniStudio utilizes this convenience.
Dynamically Set the Property
Property can be dynamically set in the program running process. It can be
implemented through setProperty
(inherited and implemented by its child
class) of mComponentClass
(this is visible component, which is basic class of
control and invisible component). Similarly, there is getProperty
method,
which is used to get value of given property. Prototypes of the two methods are
as below:
BOOL (*setProperty) (clss *self, int id, DWORD value);
DWORD (*getProperty) (clss *self, int id);
It needs to be noted that all the property values are defined as DWORD
type,
which is a 32 bit value, and can be used to store sentence handle, pointer and
integer value etc.
In the whole example program, we create a timer to set the current location property of the progress bar periodically. In the processing function of the timer, the following code calling is done:
static void mymain_onTimer (mWidget *_this, int id, DWORD count)
{
static int pb_pos = 0;
mProgressBar *pb = (mProgressBar*)ncsGetChildObj (_this->hwnd, ID_PROG);
if (pb)
_c(pb)->setProperty(pb, NCSP_PROG_CURPOS, pb_pos++);
}
Use Renderer
Renderer is a big feature of the new control set. Its main functions are:
- On the foundation of not changing the realization of internal logic of the control, customize the look and feel of the control
- Renderers change along with the variety changes of the controls, and adding and deleting a control will not generate effect on other controls, in this respect, they are different from the renderers of the old control set
As users of control, developers don’t need to implement a complete renderer, but only need to use the due renderer. The above example uses the method similar to window template to define information of the renderers used by a control.
static NCS_RDR_ELEMENT btn_rdr_elements[] =
{
{ NCS_MODE_USEFLAT, 1},
{ -1, 0 }
};
static NCS_RDR_INFO btn1_rdr_info[] =
{
{"fashion","fashion", btn_rdr_elements}
};
static NCS_RDR_INFO btn2_rdr_info[] =
{
{"flat", "flat", NULL}
};
btn_rdr_elements
defines property array of the renderer. Property of renderer is similar to the property of control, and the mechanisms are completely the same. It has {-1,0} as the end of the array, andNCS_MODE_USEFLAT
property points out that fashion renderer uses flat style (comparative, with obvious stereoscopic impression under 3D style) to draw an object.NCS_RDR_INFO
of btn1_rdr_info points out the types of renderers used by the control and their property list. Definition ofNCS_RDR_INFO
is as follows:
/**
* A renderer element struct
*
* \note This struct is used by \ref NCS_RDR_INFO when create a NCS widget
*
* \sa NCS_RDR_INFO
*/
typedef struct _NCS_RDR_ELEMENT {
/**
* The renderer element id
*/
int id;
/**
* The remderer element value
*/
DWORD value;
}NCS_RDR_ELEMENT;
/**
* A struct used to include all the renderer info when create a widget
*
* \sa NCS_CREATE_INFO, NCS_MAIN_CREATE_INFO, NCS_WND_TEMPLATE,
* NCS_MAINWND_TEMPLATE
*/
typedef struct _NCS_RDR_INFO {
/**
* The global renderer, the minigui 3.0 defined renderer
*/
const char* glb_rdr;
/**
* The control renderer, defined by NCS
*/
const char* ctl_rdr;
/**
* The Renderer element array, end by { -1, 0}
*/
NCS_RDR_ELEMENT* elements;
}NCS_RDR_INFO;
Among them,
glb_rdr
refers to global renderer, mainly used for the drawing style of non client areas. MiniGUI is responsible for drawing these areas.ctrl_rdr
refers to control renderer, mainly used for the drawing of control itself.
NCS_RDR_INFO
can be given in the template, thus control using given renderer
can be created directly.
Event Listening and Connection
mGNCS provides a signal and slot mechanism similar to QT
, which is able to
link the event of an object to any object.
Basic Principle
Event listening link mechanism of mGNCS provides a mode to decouple the event sender and event observer. It corresponds to the relation between the sender and the receiver through global data sheet. As shown in the figure below:
Figure 3 Event Listening and Connection
When an object is deleted:
- All the connections listening to the object will be deleted
- Connections that the object listens to will also be deleted
Main Interface Functions
Interfaces adding event listener are given below:
/**
* \typedef typedef BOOL (*NCS_CB_ONOBJEVENT) (mObject* self, mObject *sender,
int event_id, DWORD param);
* \brief The callback of connection
*
* \note For widgets, only support notification
*
* \param self The listener pointer
* \param sender The sender of event
* \param event_id the id of event
* \param param the param of event
*
* \return TRUE - continue dispatch the event; FALSE - stop the event.
*/
typedef BOOL (*NCS_CB_ONOBJEVENT)(mObject* self, mObject *sender,
int event_id, DWORD param);
#define NCS_CB_ONPIECEEVENT NCS_CB_ONOBJEVENT
/**
* \fn BOOL ncsAddEventListener(mObject *sender, \
* mObject* listener, \
* NCS_CB_ONOBJEVENT event_proc, \
* int event_id);
* \brief connect sender object to listener object
*
* \param sender The sender of event
* \param listener The listener of event
* \param event_proc The processing callback of event
* \param event_id The id of event
*
* \return TRUE - Sucessed, FALSE - Failed
*
* \sa ncsAddEventListener, ncsAddEventListeners, NCS_CB_ONOBJEVENT
*/
BOOL ncsAddEventListener(mObject *sender,
mObject* listener,
NCS_CB_ONOBJEVENT event_proc,
int event_id);
/**
* BOOL ncsAddEventListeners(mObject *sender,\
* mObject* listener,\
* NCS_CB_ONOBJEVENT event_proc,\
* int* event_ids);
* \brief connect a group of events from sender to listener
*
* \param sender The sender of events
* \param listener The listener of events
* \param event_proc The processing callback of events
* \param event_ids The id array of events, end by 0
*
* \return TRUE - Sucessed, FALSE - Failed
*
* \sa ncsAddEventListener, ncsAddEventListeners, NCS_CB_ONOBJEVENT
*/
BOOL ncsAddEventListeners(mObject *sender,
mObject* listener,
NCS_CB_ONOBJEVENT event_proc,
int* event_ids);
Among them,
- Callback
NCS_CB_ONOBJEVENT
defines the processing function of event, which can get object pointers of the sender and receiver simultaneously, making it unnecessary for the users to find ncsAddEventListeners
adds listening of an object to a group of events of another objectncsAddEventListener
adds listening of an object to a single event of another object
Interfaces deleting event listener are given below:
/**
* \fn BOOL ncsRemoveEventListener(mObject * listener);
* \brief remove the connections which are listened by the object
*
* \note this function is called when a listener is destroied
*
* \param listener the object pointer of listener
*
* \return TRUE - sucessed; FALSE - failed
*/
BOOL ncsRemoveEventListener(mObject * listener);
/**
* \fn BOOL ncsRemoveEventSource(mObject *sender);
* \brief remove the connections when a sender is destroied
*
* \note this function is called when a sender is destroied
*
* \param sender the sender of events
*
* \return TRUE - sucessed; FALSE - failed
*/
BOOL ncsRemoveEventSource(mObject *sender);
Among them,
- Delete a listener or event sender. Because as long as either listener or sender does not exist, the whole connection does not have the sense of existence and will be deleted.
- The two functions are automatically called by
mOBject
, and normally active calling is not needed
Interfaces exciting an event connection are given below:
/**
* \fn void ncsRaiseEvent (mObject *sender, int event_id, DWORD param);
* \biref Raise an event to listeners
*
* \note It will call the callback of \ref NCS_CB_ONOBJEVENT
* added by \ref ncsAddEventListeners, or \ref ncsAddEventListener
*
* \param sender event sender
* \param event_id the id of event
* \param param the event param
*
* \sa NCS_CB_ONOBJEVENT, ncsAddEventListener, ncsAddEventListeners
*/
void ncsRaiseEvent(mObject *sender, int event_id, DWORD param);
The function is mainly for internal use, which is used for exciting an event connection. The interface is only used when the user hopes to manually and forcedly initiate event or compile NCS control.
Usage and Example
Based on window template example program, we will add the following two event listeners:
- When clicking the Stop button, automatic update of the progress bar will be stopped through shutting off the timer;
- When clicking the Exit button, the whole dialog box will be closed.
Effect of the window created by the program is as shown in the figure below:
Figure 4 Example of Listening Event
The following codes define two listeners, which are used for Stop button and Exit button respectively:
static BOOL mymain_onExit (mMainWnd *self, mButton *sender, int id, DWORD param)
{
mymain_onClose ((mWidget*)self, MSG_CLOSE);
return TRUE;
}
static BOOL mymain_onStop (mMainWnd *self, mButton *sender, int id, DWORD param)
{
KillTimer (self->hwnd, 100);
return TRUE;
}
When the window is initialized, the two event listeners are connected to the clicking events of the above two controls:
static BOOL mymain_onCreate (mWidget* self, DWORD addData)
{
mButton *btn;
btn = (mButton*)_c(self)->getChild (self, IDC_EXIT);
ncsAddEventListener ((mObject*)btn, (mObject*)self,
(NCS_CB_ONOBJEVENT)mymain_onExit, NCSN_WIDGET_CLICKED);
btn = (mButton*)_c(self)->getChild (self, IDC_STOP);
ncsAddEventListener ((mObject*)btn, (mObject*)self,
(NCS_CB_ONOBJEVENT)mymain_onStop, NCSN_WIDGET_CLICKED);
SetTimer (self->hwnd, 100, 20);
return TRUE;
}
static void mymain_onTimer (mWidget *self, int id, DWORD count)
{
static int pb_pos = 0;
mProgressBar *pb = (mProgressBar*)ncsGetChildObj (self->hwnd, IDC_PROG);
if (pb)
_c(pb)->setProperty (pb, NCSP_PROG_CURPOS, pb_pos++);
}
static void mymain_onClose (mWidget* self, int message)
{
DestroyMainWindow (self->hwnd);
PostQuitMessage (self->hwnd);
}
static NCS_EVENT_HANDLER mymain_handlers[] = {
{MSG_CREATE, mymain_onCreate},
{MSG_CLOSE, mymain_onClose},
{MSG_TIMER, mymain_onTimer},
{0, NULL}
};
Obviously, event listener mechanism brings more convenience to the programming of the new control set, and it is convenient for miniStudio tool to implement visual programming.
Data Binding and Data Source
Data binding and data source are two very important mechanisms that mGNCS provides to the applications, which are helpful for realizing separation between the program logic and the data that it processes, and it is convenient for visual GUI design tools such as miniStudio to design interface. This chapter will simply explain the basic concepts of data binding and data source, and the next chapter will explain related interfaces and usages in detail.
If the reader ever used Visual Studio to develop C++ applications based on
MFC
, he must remember that the tool provides a mechanism, which can bind the
control content in the dialog box with the given class member variable. In
VC
, initialization value of the dialog box control can be automatically
obtained and set from the bound dialog box member variable, while when the
dialog box exits, value in the control can be automatically assigned to
corresponding dialog box class member variable. This is the idea source of data
source and data binding functions provided by mGNCS. However, the data source
and data binding functions provided by mGNCS are stronger. Utilizing data
binding function of mGNCS, when the value of mGNCS control changes, we can
automatically update other controls or store the data into the expected data
source again. Through the data source of mGNCS, we can define data source of
different formats, such as character string array defined by the program, text
file, configuration file, even database etc., and use these data to
automatically fill in mGNCS control.
Next, let's introduce data binding and data source functions of mGNCS respectively.
Data Binding
If we try to find the identical point between graphic user interface and character user interface, we will find that both of them fall into the category of human computer interaction, while the nature of human computer interaction is that data is converted between the mode that human can understand and the mode that computer can understand. Note that the above statement has three key words:
- Data
- Mode that human can understand
- Mode that computer can understand
We know that the nature of computing is computing of data, and data is the core of all computing, For computer, mode is format of data. Human computer interface interacts with human through displaying data of specific format, while when computer carries out internal computing, it hopes to compute in the format that it can understand. When the two formats don’t match, what shall we do? The answer is data binding. The figure below gives the model of NCS data binding:
Figure 5 Data Binding Model of mGNCS
Data binding transfers data between the graphic user interface and internal logic, and its advantages are:
- Decouple processing of graphic user interface and internal logic, making it easier for the developers to change interface
- Standardize and modularize applications, and enhance the extensibility of programs
- Simplify programming, free the programmers from tedious Get/Set operations
- Unify interfaces, which is good for tools such as miniStudio etc, to carry out visual operation and is good for user abstracting
The disadvantage is that extra expenditure is needed. However, codes of the operation interface themselves will bring expenditure. With the two balanced out, using data binding is very worthwhile.
List 5 gives a simple data binding example program, and interface of the program is as shown in the figure below:
Figure 6 Data Binding Example Program
The program binds the content in the edit box with the location of the track bar together:
- Drag the track bar, and the content in the edit box will change automatically to reflect the current track bar location;
- Key an integer value (0-20) in the edit box, and the current location of the track bar will change correspondingly.
List 5 data_binding.c
/*
** data_binding.c: Sample program for mGNCS Programming Guide
** Using data binding.
**
** Copyright (C) 2009 ~ 2019 FMSoft Technologies.
*/
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <minigui/common.h>
#include <minigui/minigui.h>
#include <minigui/gdi.h>
#include <minigui/window.h>
#include <mgncs/mgncs.h>
#define IDC_TRACKBAR 101
#define IDC_SLEDIT 102
// START_OF_BINDING
static BOOL mymain_onCreate (mWidget* self, DWORD add_data)
{
mTrackbar * tb = (mTrackbar*)_c(self)->getChild (self, IDC_TRACKBAR);
mSlEdit * se = (mSlEdit*) _c(self)->getChild (self, IDC_SLEDIT);
ncsConnectBindProps (NCS_CMPT_PROP (tb, NCSN_TRKBAR_CHANGED,
NCSP_TRKBAR_CURPOS, NCS_BT_INT,
NCS_PROP_FLAG_READ|NCS_PROP_FLAG_WRITE),
NCS_CMPT_PROP (se, NCSN_EDIT_CHANGE,
NCSP_WIDGET_TEXT, NCS_BT_STR,
NCS_PROP_FLAG_READ|NCS_PROP_FLAG_WRITE),
NCS_BPT_DBL);
ncsAutoReflectObjectBindProps ((mObject*)se);
return TRUE;
}
// END_OF_BINDING
static void mymain_onClose (mWidget* self, int message)
{
DestroyMainWindow (self->hwnd);
PostQuitMessage (self->hwnd);
}
static NCS_EVENT_HANDLER mymain_handlers[] = {
{MSG_CREATE, mymain_onCreate },
{MSG_CLOSE, mymain_onClose },
{0, NULL }
};
static NCS_PROP_ENTRY trackbar_props [] = {
{NCSP_TRKBAR_MINPOS, 0},
{NCSP_TRKBAR_MAXPOS, 20},
{NCSP_TRKBAR_CURPOS, 10},
{NCSP_TRKBAR_LINESTEP, 2},
{NCSP_TRKBAR_PAGESTEP, 5},
{0, 0}
};
static NCS_RDR_INFO trackbar_rdr_info[] =
{
{"flat", "flat", NULL},
};
static NCS_WND_TEMPLATE _ctrl_templ[] = {
{
NCSCTRL_TRACKBAR,
IDC_TRACKBAR,
10, 10, 240, 40,
WS_BORDER | WS_VISIBLE | NCSS_TRKBAR_NOTICK | NCSS_NOTIFY,
WS_EX_TRANSPARENT,
"",
trackbar_props,
trackbar_rdr_info,
NULL, NULL, 0, 0,
MakeRGBA(255,0,0,255)
},
{
NCSCTRL_SLEDIT,
IDC_SLEDIT,
10, 60, 240, 30,
WS_BORDER | WS_VISIBLE | NCSS_EDIT_CENTER | NCSS_NOTIFY,
WS_EX_NONE,
"edit",
NULL, NULL, NULL, NULL, 0, 0
},
};
static NCS_MNWND_TEMPLATE mymain_templ = {
NCSCTRL_DIALOGBOX,
1,
0, 0, 268, 130,
WS_CAPTION | WS_BORDER | WS_VISIBLE,
WS_EX_NONE,
"Data Binding",
NULL,
NULL,
mymain_handlers,
_ctrl_templ,
sizeof(_ctrl_templ)/sizeof(NCS_WND_TEMPLATE),
0,
MakeRGBA(255,255,255,255)
};
int MiniGUIMain(int argc, const char* argv[])
{
mDialogBox* mydlg;
if (argc > 1) {
trackbar_rdr_info[0].glb_rdr = argv[1];
trackbar_rdr_info[0].ctl_rdr = argv[1];
}
ncsInitialize ();
mydlg = (mDialogBox *)ncsCreateMainWindowIndirect
(&mymain_templ, HWND_DESKTOP);
_c(mydlg)->doModal (mydlg, TRUE);
MainWindowThreadCleanup (mydlg->hwnd);
ncsUninitialize ();
return 0;
}
Finish the above data binding, and the program only does the setting work as shown below when the dialog box is created:
static BOOL mymain_onCreate (mWidget* self, DWORD add_data)
{
mTrackbar * tb = (mTrackbar*)_c(self)->getChild (self, IDC_TRACKBAR);
mSlEdit * se = (mSlEdit*) _c(self)->getChild (self, IDC_SLEDIT);
ncsConnectBindProps (NCS_CMPT_PROP (tb, NCSN_TRKBAR_CHANGED,
NCSP_TRKBAR_CURPOS, NCS_BT_INT,
NCS_PROP_FLAG_READ|NCS_PROP_FLAG_WRITE),
NCS_CMPT_PROP (se, NCSN_EDIT_CHANGE,
NCSP_WIDGET_TEXT, NCS_BT_STR,
NCS_PROP_FLAG_READ|NCS_PROP_FLAG_WRITE),
NCS_BPT_DBL);
ncsAutoReflectObjectBindProps ((mObject*)se);
return TRUE;
}
The above code segments finish two things:
- Calling
ncsConnectBindProps
function binds the current location value property (NCSP_TRKBAR_CURPOS
) of the track bar with the text property (NCSP_WIDGET_TEXT
) of the edit box. And binding is triggered when the location generated by the track bar changes the event (NCSN_TRKBAR_CHANGED
) or the content generated by the edit box changes the event (NCSN_EDIT_CHANGE). - Calling
ncsAutoReflectObjectBindProps
function enables the edit box to automatically respond to the content change brought by data binding.
Assuming that there is no data binding function, we need to compile two event handlers to correspond to the two events of the two controls respectively, while every thing becomes very simple now.
It needs to be noted that location value property of the track bar is integer type, while the content property of the edit box is character string type, and data type conversion between the two will be automatically completed by mGNCS.
Data Source
Data source is the set of data. Through abstract data source interface, we can provide good data exchange mechanism for large controls such as list box, list type etc. The sense of defining abstract data source interface is:
- Manage data in a unified way, and it is separation between interface and data
- Unify data accessing interface, making it convenient for the development and maintenance of programs and for miniStudio tool to carry out visual processing
At present, mGNCS implements supports to the following types of data sources:
- C language data defined by applications, mGNCS refers to it as static data source;
- Text data from MiniGUI configuration file format
- Text field data with row as the unit from files such as
UNIX
passwd.
Codes in list p1c2-6 uses the static data defined by C program to initialize a list item control, as shown in the figure below:
Figure 7 Data Source Example Program
List 6 data_source.c
/*
** data_source.c: Sample program for mGNCS Programming Guide
** Using static data source.
**
** Copyright (C) 2009 ~ 2019 FMSoft Technologies.
*/
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <minigui/common.h>
#include <minigui/minigui.h>
#include <minigui/gdi.h>
#include <minigui/window.h>
#include <mgncs/mgncs.h>
#define IDC_LISTVIEW 100
// START_OF_DEFINEDS
static const NCS_LISTV_CLMRD _header[] = {
{"Name", "", 100, NCSF_LSTCLM_LEFTALIGN},
{"Sex", "", 80, NCSF_LSTCLM_LEFTALIGN},
{"Age", "", 80, NCSF_LSTCLM_LEFTALIGN}
};
static const char* _content[][3] = {
{"Jime", "Male", "15"},
{"Lily", "Female", "12"},
{"Tom", "Male", "11"}
};
// END_OF_DEFINEDS
// START_OF_SETDATA
static BOOL mymain_onCreate (mWidget* self, DWORD add_data)
{
mListView *lv = (mListView*)_c(self)->getChild (self, IDC_LISTVIEW);
if (lv) {
mRecordSet *rs;
rs = _c(g_pStaticDS)->selectRecordSet (g_pStaticDS,
"/listview/header", NCS_DS_SELECT_READ);
_c(lv)->setSpecial (lv, NCSID_LISTV_HDR, (DWORD)rs, NULL);
rs = _c(g_pStaticDS)->selectRecordSet (g_pStaticDS,
"/listview/content", NCS_DS_SELECT_READ);
_c(lv)->setSpecial (lv, NCS_CONTENT, (DWORD)rs, NULL);
}
return TRUE;
}
// END_OF_SETDATA
static void mymain_onClose (mWidget* self, int message)
{
DestroyMainWindow (self->hwnd);
PostQuitMessage (self->hwnd);
}
static NCS_EVENT_HANDLER mymain_handlers[] = {
{MSG_CREATE, mymain_onCreate},
{MSG_CLOSE, mymain_onClose},
{0, NULL}
};
static NCS_WND_TEMPLATE _ctrl_templ[] = {
{
NCSCTRL_LISTVIEW,
IDC_LISTVIEW,
10, 10, 320, 190,
WS_BORDER | WS_VISIBLE | NCSS_LISTV_SORT
| NCSS_LISTV_LOOP,
WS_EX_NONE,
"",
NULL, NULL, NULL, NULL, 0, 0
}
};
static NCS_MNWND_TEMPLATE mymain_templ = {
NCSCTRL_MAINWND,
1,
0, 0, 350, 240,
WS_CAPTION | WS_BORDER | WS_VISIBLE,
WS_EX_NONE,
"Data Source ....",
NULL,
NULL,
mymain_handlers,
_ctrl_templ,
sizeof(_ctrl_templ)/sizeof(NCS_WND_TEMPLATE),
0, 0, 0,
};
int MiniGUIMain (int argc, const char* argv[])
{
mDialogBox* mydlg;
ncsInitialize ();
// START_OF_REGDS
ncsRegisterStaticData ("/listview/header", (void*)_header, 3,
sizeof(NCS_LISTV_CLMRD)/sizeof(DWORD), sizeof(DWORD),
gListVColumnRecordTypes);
ncsRegisterStaticData ("/listview/content", (void*)_content, 3,
3, sizeof(char*), NULL);
// END_OF_REGDS
mydlg = (mDialogBox *)ncsCreateMainWindowIndirect
(&mymain_templ, HWND_DESKTOP);
_c(mydlg)->doModal (mydlg, TRUE);
MainWindowThreadCleanup (mydlg->hwnd);
ncsUninitialize ();
return 0;
}
To implement the above data source function, the program completes work in the following three aspects:
First step, define data source:
static const NCS_LISTV_CLMRD _header[] = {
{"Name", "", 100, NCSF_LSTCLM_LEFTALIGN},
{"Sex", "", 80, NCSF_LSTCLM_LEFTALIGN},
{"Age", "", 80, NCSF_LSTCLM_LEFTALIGN}
};
static const char* _content[][3] = {
{"Jime", "Male", "15"},
{"Lily", "Female", "12"},
{"Tom", "Male", "11"}
};
The above codes define the header information and content data of list control
Second step, Register data source:
ncsRegisterStaticData ("/listview/header", (void*)_header, 3,
sizeof(NCS_LISTV_CLMRD)/sizeof(DWORD), sizeof(DWORD),
gListVColumnRecordTypes);
ncsRegisterStaticData ("/listview/content", (void*)_content, 3,
3, sizeof(char*), NULL);
The above codes define the two kinds of data into “listview/header” and “listview/content”. When it is necessary to quote the two kinds of data, just use the names given here.
Third step, use data source
static BOOL mymain_onCreate (mWidget* self, DWORD add_data)
{
mListView *lv = (mListView*)_c(self)->getChild (self, IDC_LISTVIEW);
if (lv) {
mRecordSet *rs;
rs = _c(g_pStaticDS)->selectRecordSet (g_pStaticDS,
"/listview/header", NCS_DS_SELECT_READ);
_c(lv)->setSpecial (lv, NCSID_LISTV_HDR, (DWORD)rs, NULL);
rs = _c(g_pStaticDS)->selectRecordSet (g_pStaticDS,
"/listview/content", NCS_DS_SELECT_READ);
_c(lv)->setSpecial (lv, NCS_CONTENT, (DWORD)rs, NULL);
}
return TRUE;
}
The above codes get corresponding data from the data source and then call
setSpecificData
method of list control to set.
Brief Summary
Through study of this chapter, we can have an overall understanding of mGNCS. Compared to MiniGUI intrinsic control system, mGNCS reconstructs the control system of MiniGUI, making it more standardized and easier for customization. In addition, this design adapts to miniStudio integrated development environment of MiniGUI application development, and relation between the two is mutually complementing.
Compared to the intrinsic control system of MiniGUI, mGNCS introduces object-oriented design idea, greatly enhancing the customizability and extensibility of MiniGUI, and making it convenient for visual design. Of course, for application development personnel, it brings longer study path, but I believe that through explanation of this programming guide, any development personnel mastering MiniGUI application development can rapidly master mGNCS. Of course, the original intention of designing and introducing mGNCS is to develop better design tool, which is miniStudio, while most of the functions can be obtained through visual design in miniStudio. Undoubtedly, using mGNCS matched with miniStudio, development efficiency of embedded graphic interface applications will be greatly enhanced.
<< Brief Introduction | Table of Contents | Renderer and Resources Management >>