mirror of
https://github.com/tuxbox-neutrino/neutrino.git
synced 2025-08-28 16:01:20 +02:00
795 lines
24 KiB
C++
795 lines
24 KiB
C++
/*
|
|
Based up Neutrino-GUI - Tuxbox-Project
|
|
Copyright (C) 2001 by Steffen Hehn 'McClean'
|
|
|
|
Classes for generic GUI-related components.
|
|
Copyright (C) 2012, 2013, 2014 Thilo Graf 'dbt'
|
|
Copyright (C) 2012, Michael Liebmann 'micha-bbg'
|
|
|
|
License: GPL
|
|
|
|
This program is free software; you can redistribute it and/or
|
|
modify it under the terms of the GNU General Public
|
|
License as published by the Free Software Foundation; either
|
|
version 2 of the License, or (at your option) any later version.
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
#include <config.h>
|
|
#endif
|
|
|
|
#include <global.h>
|
|
#include <neutrino.h>
|
|
#include "cc_frm.h"
|
|
#include <stdlib.h>
|
|
#include <algorithm>
|
|
#include <system/debug.h>
|
|
#include <stdexcept>
|
|
|
|
using namespace std;
|
|
|
|
//-------------------------------------------------------------------------------------------------------
|
|
//sub class CComponentsForm from CComponentsItem
|
|
CComponentsForm::CComponentsForm( const int x_pos, const int y_pos, const int w, const int h,
|
|
CComponentsForm* parent,
|
|
int shadow_mode,
|
|
fb_pixel_t color_frame,
|
|
fb_pixel_t color_body,
|
|
fb_pixel_t color_shadow)
|
|
:CComponentsItem(parent)
|
|
{
|
|
cc_item_type.id = CC_ITEMTYPE_FRM;
|
|
cc_item_type.name= "cc_base_form";
|
|
|
|
Init(x_pos, y_pos, w, h, color_frame, color_body, color_shadow);
|
|
cc_xr = x;
|
|
cc_yr = y;
|
|
|
|
shadow = shadow_mode;
|
|
shadow_w = OFFSET_SHADOW;
|
|
corner_rad = RADIUS_LARGE;
|
|
corner_type = CORNER_ALL;
|
|
cc_item_index = 0;
|
|
|
|
//add default exit keys for exec handler
|
|
v_exit_keys.push_back(CRCInput::RC_home);
|
|
v_exit_keys.push_back(CRCInput::RC_setup);
|
|
|
|
v_cc_items.clear();
|
|
|
|
append_x_offset = 0;
|
|
append_y_offset = 0;
|
|
page_count = 1;
|
|
cur_page = 0;
|
|
sb = NULL;
|
|
w_sb = SCROLLBAR_WIDTH;
|
|
|
|
page_scroll_mode = PG_SCROLL_M_UP_DOWN_KEY;
|
|
|
|
//connect page scroll slot
|
|
sigc::slot4<void, neutrino_msg_t&, neutrino_msg_data_t&, int&, bool&> sl = sigc::mem_fun(*this, &CComponentsForm::execPageScroll);
|
|
this->OnExec.connect(sl);
|
|
}
|
|
|
|
void CComponentsForm::Init( const int& x_pos, const int& y_pos, const int& w, const int& h,
|
|
const fb_pixel_t& color_frame,
|
|
const fb_pixel_t& color_body,
|
|
const fb_pixel_t& color_shadow)
|
|
{
|
|
setDimensionsAll(x_pos, y_pos, w, h);
|
|
setColorAll(color_frame, color_body, color_shadow);
|
|
}
|
|
|
|
CComponentsForm::~CComponentsForm()
|
|
{
|
|
clear();
|
|
delete sb;
|
|
}
|
|
|
|
int CComponentsForm::exec()
|
|
{
|
|
dprintf(DEBUG_NORMAL, "[CComponentsForm] [%s - %d] \n", __func__, __LINE__);
|
|
|
|
//basic values
|
|
neutrino_msg_t msg;
|
|
neutrino_msg_data_t data;
|
|
int res = menu_return::RETURN_REPAINT;
|
|
uint64_t timeoutEnd = CRCInput::calcTimeoutEnd(-1);
|
|
|
|
//allow exec loop
|
|
bool cancel_exec = false;
|
|
|
|
//signal before exec
|
|
OnBeforeExec();
|
|
|
|
while (!cancel_exec)
|
|
{
|
|
g_RCInput->getMsgAbsoluteTimeout( &msg, &data, &timeoutEnd );
|
|
|
|
//execute connected slots
|
|
OnExec(msg, data, res, cancel_exec);
|
|
//exit loop
|
|
execExit(msg, data, res, cancel_exec, v_exit_keys);
|
|
|
|
if (CNeutrinoApp::getInstance()->handleMsg(msg, data) & messages_return::cancel_all)
|
|
{
|
|
dprintf(DEBUG_INFO, "[CComponentsForm] [%s - %d] messages_return::cancel_all\n", __func__, __LINE__);
|
|
res = menu_return::RETURN_EXIT_ALL;
|
|
cancel_exec = EXIT;
|
|
}
|
|
}
|
|
|
|
//signal after exec
|
|
OnAfterExec();
|
|
|
|
return res;
|
|
}
|
|
|
|
|
|
void CComponentsForm::execKey(neutrino_msg_t& msg, neutrino_msg_data_t& data, int& res, bool& cancel_exec, const std::vector<neutrino_msg_t>& v_msg_list, bool force_exit)
|
|
{
|
|
for(size_t i = 0; i < v_msg_list.size(); i++){
|
|
if (execKey(msg, data, res, cancel_exec, v_msg_list[i], force_exit)){
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
inline bool CComponentsForm::execKey(neutrino_msg_t& msg, neutrino_msg_data_t& data, int& res, bool& cancel_exec, const neutrino_msg_t& msg_val, bool force_exit)
|
|
{
|
|
if (msg == msg_val){
|
|
OnExecMsg(msg, data, res);
|
|
if (force_exit)
|
|
cancel_exec = EXIT;
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
|
|
void CComponentsForm::execPageScroll(neutrino_msg_t& msg, neutrino_msg_data_t& /*data*/, int& /*res*/, bool& /*cancel_exec*/)
|
|
{
|
|
if (page_scroll_mode == PG_SCROLL_M_OFF)
|
|
return;
|
|
|
|
if (page_scroll_mode & PG_SCROLL_M_UP_DOWN_KEY){
|
|
if (msg == CRCInput::RC_page_up)
|
|
ScrollPage(SCROLL_P_DOWN);
|
|
if (msg == CRCInput::RC_page_down)
|
|
ScrollPage(SCROLL_P_UP);
|
|
}
|
|
|
|
if (page_scroll_mode & PG_SCROLL_M_LEFT_RIGHT_KEY){
|
|
if (msg == CRCInput::RC_left)
|
|
ScrollPage(SCROLL_P_DOWN);
|
|
if (msg == CRCInput::RC_right)
|
|
ScrollPage(SCROLL_P_UP);
|
|
}
|
|
}
|
|
|
|
void CComponentsForm::execExit(neutrino_msg_t& msg, neutrino_msg_data_t& data, int& res, bool& cancel_exec, const std::vector<neutrino_msg_t>& v_msg_list)
|
|
{
|
|
execKey(msg, data, res, cancel_exec, v_msg_list, true);
|
|
}
|
|
|
|
|
|
void CComponentsForm::clear()
|
|
{
|
|
if (v_cc_items.empty())
|
|
return;
|
|
|
|
std::lock_guard<std::mutex> g(cc_frm_mutex);
|
|
|
|
for(size_t i=0; i<v_cc_items.size(); i++) {
|
|
if (v_cc_items.at(i)){
|
|
dprintf(DEBUG_DEBUG, "[CComponentsForm]\t[%s-%d] delete form cc-item %d of %d address = %p \033[33m\t type = [%d] [%s]\033[0m\n", __func__, __LINE__, (int)i+1, (int)v_cc_items.size(), v_cc_items.at(i), v_cc_items.at(i)->getItemType(), v_cc_items.at(i)->getItemName().c_str());
|
|
delete v_cc_items.at(i);
|
|
v_cc_items.at(i) = NULL;
|
|
}
|
|
}
|
|
v_cc_items.clear();
|
|
}
|
|
|
|
|
|
int CComponentsForm::addCCItem(CComponentsItem* cc_Item)
|
|
{
|
|
if (cc_Item){
|
|
std::lock_guard<std::mutex> g(cc_frm_mutex);
|
|
dprintf(DEBUG_DEBUG, "[CComponentsForm]\t[%s-%d] try to add cc_Item \033[33m\t type = [%d] [%s]\033[0m to form [%s -> current index=%d] \n", __func__, __LINE__, cc_Item->getItemType(), cc_Item->getItemName().c_str(), getItemName().c_str(), cc_item_index);
|
|
|
|
cc_Item->setParent(this);
|
|
v_cc_items.push_back(cc_Item);
|
|
|
|
dprintf(DEBUG_DEBUG, "\tadded cc_Item (type = [%d] [%s] to form [current index=%d] \n", cc_Item->getItemType(), cc_Item->getItemName().c_str(), cc_item_index);
|
|
|
|
//assign item index
|
|
int new_index = genIndex();
|
|
cc_Item->setIndex(new_index);
|
|
cc_Item->setFocus(true);
|
|
|
|
dprintf(DEBUG_DEBUG, "\t%s-%d parent index = %d, assigned index ======> %d\n", __func__, __LINE__, cc_item_index, new_index);
|
|
return getCCItemId(cc_Item);
|
|
}
|
|
else
|
|
dprintf(DEBUG_NORMAL, "[CComponentsForm] %s-%d tried to add an empty or invalide cc_item !!!\n", __func__, __LINE__);
|
|
return -1;
|
|
}
|
|
|
|
int CComponentsForm::addCCItem(const std::vector<CComponentsItem*> &cc_Items)
|
|
{
|
|
for (size_t i= 0; i< cc_Items.size(); i++)
|
|
addCCItem(cc_Items.at(i));
|
|
return size();
|
|
}
|
|
|
|
int CComponentsForm::getCCItemId(CComponentsItem* cc_Item) const
|
|
{
|
|
if (cc_Item){
|
|
for (size_t i= 0; i< v_cc_items.size(); i++)
|
|
if (v_cc_items.at(i) == cc_Item)
|
|
return i;
|
|
}
|
|
return -1;
|
|
}
|
|
|
|
int CComponentsForm::genIndex()
|
|
{
|
|
int count = v_cc_items.size();
|
|
char buf[64];
|
|
snprintf(buf, sizeof(buf), "%d%d", cc_item_index, count);
|
|
buf[63] = '\0';
|
|
int ret = atoi(buf);
|
|
return ret;
|
|
}
|
|
|
|
CComponentsItem* CComponentsForm::getCCItem(const uint& cc_item_id) const
|
|
{
|
|
if (cc_item_id >= size()){
|
|
dprintf(DEBUG_NORMAL, "[CComponentsForm] [%s - %d] Error: inside container type = [%d] [%s] parameter cc_item_id = %u, out of range (size = %zx)...\n", __func__, __LINE__, cc_item_type.id, cc_item_type.name.c_str(), cc_item_id, size());
|
|
return NULL;
|
|
}
|
|
|
|
if (v_cc_items.at(cc_item_id))
|
|
return v_cc_items.at(cc_item_id);
|
|
return NULL;
|
|
}
|
|
|
|
CComponentsItem* CComponentsForm::getPrevCCItem(CComponentsItem* current_cc_item) const
|
|
{
|
|
return getCCItem(getCCItemId(current_cc_item) - 1);
|
|
}
|
|
|
|
CComponentsItem* CComponentsForm::getNextCCItem(CComponentsItem* current_cc_item) const
|
|
{
|
|
return getCCItem(getCCItemId(current_cc_item) + 1);
|
|
}
|
|
|
|
void CComponentsForm::replaceCCItem(const uint& cc_item_id, CComponentsItem* new_cc_Item)
|
|
{
|
|
if (!v_cc_items.empty()){
|
|
|
|
std::lock_guard<std::mutex> g(cc_frm_mutex);
|
|
CComponentsItem* old_Item = v_cc_items.at(cc_item_id);
|
|
if (old_Item){
|
|
CComponentsForm * old_parent = old_Item->getParent();
|
|
new_cc_Item->setParent(old_parent);
|
|
new_cc_Item->setIndex(old_parent->getIndex());
|
|
delete old_Item;
|
|
old_Item = NULL;
|
|
v_cc_items.at(cc_item_id) = new_cc_Item;
|
|
}
|
|
}
|
|
else
|
|
dprintf(DEBUG_NORMAL, "[CComponentsForm] %s replace cc_Item not possible, v_cc_items is empty\n", __func__);
|
|
}
|
|
|
|
void CComponentsForm::replaceCCItem(CComponentsItem* old_cc_Item, CComponentsItem* new_cc_Item)
|
|
{
|
|
replaceCCItem(getCCItemId(old_cc_Item), new_cc_Item);
|
|
}
|
|
|
|
void CComponentsForm::insertCCItem(const uint& cc_item_id, CComponentsItem* cc_Item)
|
|
{
|
|
|
|
if (cc_Item == NULL){
|
|
dprintf(DEBUG_DEBUG, "[CComponentsForm] %s parameter: cc_Item = %p...\n", __func__, cc_Item);
|
|
return;
|
|
}
|
|
|
|
if (v_cc_items.empty()){
|
|
addCCItem(cc_Item);
|
|
dprintf(DEBUG_NORMAL, "[CComponentsForm] %s insert cc_Item not possible, v_cc_items is empty, cc_Item added\n", __func__);
|
|
}else{
|
|
|
|
std::lock_guard<std::mutex> g(cc_frm_mutex);
|
|
v_cc_items.insert(v_cc_items.begin()+cc_item_id, cc_Item);
|
|
cc_Item->setParent(this);
|
|
//assign item index
|
|
int index = genIndex();
|
|
cc_Item->setIndex(index);
|
|
}
|
|
}
|
|
|
|
void CComponentsForm::removeCCItem(const uint& cc_item_id)
|
|
{
|
|
if (!v_cc_items.empty()){
|
|
std::lock_guard<std::mutex> g(cc_frm_mutex);
|
|
if (v_cc_items.at(cc_item_id)) {
|
|
delete v_cc_items.at(cc_item_id);
|
|
v_cc_items.at(cc_item_id) = NULL;
|
|
v_cc_items.erase(v_cc_items.begin()+cc_item_id);
|
|
dprintf(DEBUG_DEBUG, "[CComponentsForm] %s removing cc_Item [id=%u]...\n", __func__, cc_item_id);
|
|
}
|
|
}
|
|
else
|
|
dprintf(DEBUG_NORMAL, "[CComponentsForm] %s removing of cc_Item [id=%u] not possible, v_cc_items is empty...\n", __func__, cc_item_id);
|
|
}
|
|
|
|
void CComponentsForm::removeCCItem(CComponentsItem* cc_Item)
|
|
{
|
|
uint id = getCCItemId(cc_Item);
|
|
removeCCItem(id);
|
|
}
|
|
|
|
void CComponentsForm::exchangeCCItem(const uint& cc_item_id_a, const uint& cc_item_id_b)
|
|
{
|
|
if (!v_cc_items.empty())
|
|
swap(v_cc_items.at(cc_item_id_a), v_cc_items.at(cc_item_id_b));
|
|
}
|
|
|
|
void CComponentsForm::exchangeCCItem(CComponentsItem* item_a, CComponentsItem* item_b)
|
|
{
|
|
exchangeCCItem(getCCItemId(item_a), getCCItemId(item_b));
|
|
}
|
|
|
|
void CComponentsForm::paintForm(const bool &do_save_bg)
|
|
{
|
|
std::mutex paint_mutex;
|
|
std::lock_guard<std::mutex> g(paint_mutex);
|
|
//paint body
|
|
if (!is_painted || force_paint_bg || shadow_force){
|
|
if (is_painted)
|
|
clearScreenBuffer(); //ensure clean screen buffers to avoid possible crash if layers were already was painted and any screen was saved before
|
|
paintInit(do_save_bg);
|
|
}
|
|
|
|
//paint
|
|
paintCCItems();
|
|
}
|
|
|
|
void CComponentsForm::paint(const bool &do_save_bg)
|
|
{
|
|
if(is_painted)
|
|
OnBeforeRePaint();
|
|
OnBeforePaint();
|
|
paintForm(do_save_bg);
|
|
}
|
|
|
|
bool CComponentsForm::isPageChanged()
|
|
{
|
|
for(size_t i=0; i<v_cc_items.size(); i++){
|
|
if (v_cc_items.at(i)) {
|
|
if (v_cc_items.at(i)->getPageNumber() != cur_page){
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
void CComponentsForm::paintPage(const uint8_t& page_number, const bool &do_save_bg)
|
|
{
|
|
cur_page = page_number;
|
|
paint(do_save_bg);
|
|
}
|
|
|
|
void CComponentsForm::paintCCItems()
|
|
{
|
|
//using of real x/y values to paint items if this text object is bound in a parent form
|
|
int this_x = x, auto_x = x, this_y = y, auto_y = y, this_w = 0;
|
|
int w_parent_frame = 0;
|
|
if (cc_parent){
|
|
this_x = auto_x = cc_xr;
|
|
this_y = auto_y = cc_yr;
|
|
w_parent_frame = cc_parent->getFrameThickness();
|
|
}
|
|
|
|
//init and handle scrollbar
|
|
getPageCount();
|
|
int y_sb = this_y;
|
|
int x_sb = this_x + width - w_sb;
|
|
int h_sb = height;
|
|
if (sb == NULL){
|
|
sb = new CComponentsScrollBar(x_sb, y_sb, w_sb, h_sb);
|
|
}else{
|
|
//clean background, if dimension of scrollbar was changed
|
|
if (w_sb != sb->getWidth())
|
|
sb->kill(col_body);
|
|
|
|
//set current dimensions and position
|
|
sb->setDimensionsAll(x_sb, y_sb, w_sb, h_sb);
|
|
}
|
|
|
|
if(page_count > 1){
|
|
sb->setSegmentCount(page_count);
|
|
sb->setMarkID(cur_page);
|
|
this_w = width - w_sb;
|
|
sb->paint(false);
|
|
}else{
|
|
if (sb->isPainted())
|
|
sb->kill(col_body);
|
|
this_w = width;
|
|
}
|
|
|
|
//detect if current page has changed, if true then kill items from screen
|
|
if(isPageChanged()){
|
|
this->killCCItems(col_body, true);
|
|
}
|
|
|
|
try{
|
|
for(size_t i=0; i<v_cc_items.size(); i++){
|
|
std::lock_guard<std::mutex> g(cc_frm_mutex);
|
|
|
|
if (!v_cc_items.at(i))
|
|
continue;
|
|
|
|
dprintf(DEBUG_DEBUG, "[CComponentsForm] %s: page_count = %u, item_page = %u, cur_page = %u\n", __func__, getPageCount(), v_cc_items.at(i)->getPageNumber(), this->cur_page);
|
|
|
|
//get current position of item
|
|
int xpos = v_cc_items.at(i)->getXPos();
|
|
int ypos = v_cc_items.at(i)->getYPos();
|
|
|
|
//get current dimension of item
|
|
int w_item = v_cc_items.at(i)->getWidth() - (xpos <= fr_thickness ? fr_thickness : 0);
|
|
int h_item = v_cc_items.at(i)->getHeight() - (ypos <= fr_thickness ? fr_thickness : 0);
|
|
/* this happens for example with opkg manager's "package info" screen on a SD framebuffer (704x576)
|
|
* later CC_CENTERED auto_y calculation is wrong then */
|
|
if (v_cc_items.at(i)->getHeight() > height) {
|
|
v_cc_items.at(i)->setHeight(height);
|
|
// TODO: the fr_thickness is probably wrong for ypos < 0 (CC_CENTERED / CC_APPEND)
|
|
h_item = v_cc_items.at(i)->getHeight() - (ypos <= fr_thickness ? fr_thickness : 0);
|
|
}
|
|
|
|
//check item for corrupt position, skip current item if found problems
|
|
if (ypos > height || xpos > this_w){
|
|
dprintf(DEBUG_INFO, "[CComponentsForm] %s: [form: %d] [item-index %d] \033[33m\t type = [%d] [%s]\033[0m WARNING: item position is out of form size:\ndefinied x=%d, defined this_w=%d \ndefinied y=%d, defined height=%d \n",
|
|
__func__, cc_item_index, v_cc_items.at(i)->getIndex(), v_cc_items.at(i)->getItemType(), v_cc_items.at(i)->getItemName().c_str(), xpos, this_w, ypos, height);
|
|
if (this->cc_item_type.id != CC_ITEMTYPE_FRM_CHAIN)
|
|
continue;
|
|
}
|
|
|
|
//move item x-position, if we have a frame on parent, TODO: other constellations not considered at the moment
|
|
w_parent_frame = xpos <= fr_thickness ? fr_thickness : w_parent_frame;
|
|
|
|
//set required x-position to item:
|
|
//append vertical
|
|
if (xpos == CC_APPEND){
|
|
auto_x += append_x_offset;
|
|
v_cc_items.at(i)->setRealXPos(auto_x + xpos + w_parent_frame);
|
|
auto_x += w_item;
|
|
}
|
|
//positionize vertical centered
|
|
else if (xpos == CC_CENTERED){
|
|
auto_x = this_w/2 - w_item/2;
|
|
v_cc_items.at(i)->setRealXPos(this_x + auto_x + w_parent_frame);
|
|
}
|
|
else{
|
|
v_cc_items.at(i)->setRealXPos(this_x + xpos + w_parent_frame);
|
|
auto_x = (v_cc_items.at(i)->getRealXPos() + w_item);
|
|
}
|
|
|
|
//move item y-position, if we have a frame on parent, TODO: other constellations not considered at the moment
|
|
w_parent_frame = ypos <= fr_thickness ? fr_thickness : w_parent_frame;
|
|
|
|
//set required y-position to item
|
|
//append hor
|
|
if (ypos == CC_APPEND){
|
|
auto_y += append_y_offset;
|
|
// FIXME: ypos is probably wrong here, because it is -1
|
|
v_cc_items.at(i)->setRealYPos(auto_y + ypos + w_parent_frame);
|
|
auto_y += h_item;
|
|
}
|
|
//positionize hor centered
|
|
else if (ypos == CC_CENTERED){
|
|
auto_y = height/2 - h_item/2;
|
|
v_cc_items.at(i)->setRealYPos(this_y + auto_y + w_parent_frame);
|
|
}
|
|
else{
|
|
v_cc_items.at(i)->setRealYPos(this_y + ypos + w_parent_frame);
|
|
auto_y = (v_cc_items.at(i)->getRealYPos() + h_item);
|
|
}
|
|
|
|
//reduce corner radius, if we have a frame around parent item, ensure matching corners inside of embedded item, this avoids ugly unpainted spaces between frame and item border
|
|
//TODO: other constellations not considered at the moment
|
|
if (w_parent_frame)
|
|
v_cc_items.at(i)->setCorner(max(0, v_cc_items.at(i)->getCornerRadius()- w_parent_frame), v_cc_items.at(i)->getCornerType());
|
|
|
|
//These steps check whether the element can be painted into the container.
|
|
//Is it too wide or too high, it will be shortened and displayed in the log.
|
|
//This should be avoid!
|
|
//checkwidth and adapt if required
|
|
int right_frm = (cc_parent ? cc_xr : x) + this_w/* - 2*fr_thickness*/;
|
|
int right_item = v_cc_items.at(i)->getRealXPos() + w_item;
|
|
int w_diff = max(0, right_item - right_frm);
|
|
int new_w = w_item - w_diff;
|
|
//avoid of width error due to odd values (1 line only)
|
|
right_item -= (new_w%2);
|
|
w_item -= (new_w%2);
|
|
if (right_item > right_frm){
|
|
dprintf(DEBUG_INFO, "[CComponentsForm] %s: [form: %d] [item-index %d] \033[33m\t type = [%d] [%s]\033[0m this_w is too large, definied width=%d, possible width=%d \n",
|
|
__func__, cc_item_index, v_cc_items.at(i)->getIndex(), v_cc_items.at(i)->getItemType(), v_cc_items.at(i)->getItemName().c_str(), w_item, new_w);
|
|
v_cc_items.at(i)->setWidth(new_w);
|
|
}
|
|
|
|
//check height and adapt if required
|
|
int bottom_frm = (cc_parent ? cc_yr : y) + height/* - 2*fr_thickness*/;
|
|
int bottom_item = v_cc_items.at(i)->getRealYPos() + h_item;
|
|
int h_diff = max(0, bottom_item - bottom_frm);
|
|
int new_h = h_item - h_diff;
|
|
//avoid of height error due to odd values (1 line only)
|
|
bottom_item -= (new_h%2);
|
|
h_item -= (new_h%2);
|
|
if (bottom_item > bottom_frm){
|
|
dprintf(DEBUG_INFO, "[CComponentsForm] %s: [form: %d] [item-index %d] \033[33m\t type = [%d] [%s]\033[0m height is too large, definied height=%d, possible height=%d \n",
|
|
__func__, cc_item_index, v_cc_items.at(i)->getIndex(), v_cc_items.at(i)->getItemType(), v_cc_items.at(i)->getItemName().c_str(), h_item, new_h);
|
|
v_cc_items.at(i)->setHeight(new_h);
|
|
}
|
|
|
|
//get current visibility mode from item, me must hold it and restore after paint
|
|
const bool item_visible = v_cc_items.at(i)->paintAllowed();
|
|
//set visibility mode
|
|
if (!cc_allow_paint)
|
|
v_cc_items.at(i)->allowPaint(false);
|
|
|
|
//restore background
|
|
if (v_cc_items.at(i)->isPainted())
|
|
v_cc_items.at(i)->hide();
|
|
|
|
//finally paint current item, but only required contents of page
|
|
if (v_cc_items.at(i)->getPageNumber() == cur_page)
|
|
v_cc_items.at(i)->paint(v_cc_items.at(i)->SaveBg());
|
|
|
|
//restore defined old visibility mode of item after paint
|
|
if (v_cc_items.at(i))
|
|
v_cc_items.at(i)->allowPaint(item_visible);
|
|
}
|
|
}
|
|
catch (const out_of_range& err0)
|
|
{
|
|
dprintf(DEBUG_NORMAL, "\033[31m[CComponentsForm]\[%s - %d], Error in item %s: Out of Range error: %s\033[0m\n", __func__, __LINE__, getItemName().c_str(), err0.what());
|
|
return;
|
|
}
|
|
catch (const std::runtime_error& err1)
|
|
{
|
|
dprintf(DEBUG_NORMAL, "\033[31m[CComponentsForm]\[%s - %d], runtime error in item %s: %s\033[0m\n", __func__, __LINE__, getItemName().c_str(), err1.what());
|
|
return;
|
|
}
|
|
catch (...)
|
|
{
|
|
dprintf(DEBUG_NORMAL, "\033[31m[CComponentsForm]\[%s - %d], unknown error in item %s: FIXME!\033[0m\n", __func__, __LINE__, getItemName().c_str());
|
|
return;
|
|
}
|
|
}
|
|
|
|
//erase or paint over rendered objects
|
|
void CComponentsForm::killCCItems(const fb_pixel_t& bg_color, bool ignore_parent)
|
|
{
|
|
|
|
std::lock_guard<std::mutex> g(cc_frm_mutex);
|
|
for(size_t i=0; i<v_cc_items.size(); i++){
|
|
if (v_cc_items.at(i))
|
|
v_cc_items.at(i)->kill(bg_color, ignore_parent);
|
|
}
|
|
}
|
|
|
|
void CComponentsForm::hideCCItems()
|
|
{
|
|
|
|
std::lock_guard<std::mutex> g(cc_frm_mutex);
|
|
for(size_t i=0; i<v_cc_items.size(); i++){
|
|
if (v_cc_items.at(i))
|
|
v_cc_items.at(i)->hide();
|
|
}
|
|
}
|
|
|
|
void CComponentsForm::setPageCount(const uint8_t& pageCount)
|
|
{
|
|
|
|
std::lock_guard<std::mutex> g(cc_frm_mutex);
|
|
|
|
uint8_t new_val = pageCount;
|
|
if (new_val < page_count)
|
|
dprintf(DEBUG_NORMAL, "[CComponentsForm] %s: current count (= %u) of pages higher than page_count (= %u) will be set, smaller value is ignored!\n", __func__, page_count, new_val) ;
|
|
page_count = max(new_val, page_count);
|
|
}
|
|
|
|
uint8_t CComponentsForm::getPageCount()
|
|
{
|
|
uint8_t num = 0;
|
|
for(size_t i=0; i<v_cc_items.size(); i++){
|
|
if (v_cc_items.at(i)){
|
|
uint8_t item_num = v_cc_items.at(i)->getPageNumber();
|
|
num = max(item_num, num);
|
|
}
|
|
}
|
|
|
|
//convert type, possible -Wconversion warnings!
|
|
page_count = static_cast<uint8_t>(num+1);
|
|
|
|
return page_count;
|
|
}
|
|
|
|
|
|
void CComponentsForm::setSelectedItem(int item_id, const fb_pixel_t& sel_frame_col, const fb_pixel_t& frame_col, const fb_pixel_t& sel_body_col, const fb_pixel_t& body_col, const int& frame_w, const int& sel_frame_w)
|
|
{
|
|
std::lock_guard<std::mutex> g(cc_frm_mutex);
|
|
|
|
size_t count = v_cc_items.size();
|
|
int id = item_id;
|
|
|
|
if (id > (int)(count-1) || id < 0 || (count == 0)){
|
|
dprintf(DEBUG_NORMAL, "[CComponentsForm] [%s - %d] invalid parameter item_id = %u, available items = %zu, allowed values are: 0...%zu! \n", __func__,
|
|
__LINE__,
|
|
item_id,
|
|
count,
|
|
count==0 ? 0:count-1);
|
|
//exit if no item is available
|
|
if (count == 0)
|
|
return;
|
|
|
|
//jump to last item
|
|
if (id < 0)
|
|
id = count-1;
|
|
//jump to 1st item, if id is out of range, avoids also possible segfault
|
|
if (id > (int)(count-1))
|
|
id = 0;
|
|
}
|
|
|
|
for (size_t i= 0; i< count; i++){
|
|
if (v_cc_items.at(i))
|
|
v_cc_items.at(i)->setSelected(i == (size_t)id, sel_frame_col, frame_col, sel_body_col, body_col, frame_w, sel_frame_w);
|
|
}
|
|
|
|
OnSelect();
|
|
}
|
|
|
|
void CComponentsForm::setSelectedItem(CComponentsItem* cc_item, const fb_pixel_t& sel_frame_col, const fb_pixel_t& frame_col, const fb_pixel_t& sel_body_col, const fb_pixel_t& body_col, const int& frame_w, const int& sel_frame_w)
|
|
{
|
|
int id = getCCItemId(cc_item);
|
|
if (id == -1){
|
|
dprintf(DEBUG_NORMAL, "[CComponentsForm] [%s - %d] invalid item parameter, no object available\n", __func__,__LINE__);
|
|
return;
|
|
}
|
|
setSelectedItem(id, sel_frame_col, frame_col, sel_body_col, body_col, frame_w, sel_frame_w);
|
|
}
|
|
|
|
int CComponentsForm::getSelectedItem() const
|
|
{
|
|
for (size_t i= 0; i< size(); i++)
|
|
if (getCCItem(i)->isSelected())
|
|
return static_cast<int>(i);
|
|
return -1;
|
|
}
|
|
|
|
CComponentsItem* CComponentsForm::getSelectedItemObject() const
|
|
{
|
|
int sel = getSelectedItem();
|
|
CComponentsItem* ret = NULL;
|
|
if (sel != -1)
|
|
ret = static_cast<CComponentsItem*>(this->getCCItem(sel));
|
|
|
|
return ret;
|
|
}
|
|
|
|
|
|
void CComponentsForm::ScrollPage(int direction, bool do_paint)
|
|
{
|
|
if (getPageCount() == 1){
|
|
cur_page = 0;
|
|
return;
|
|
}
|
|
|
|
OnBeforeScrollPage();
|
|
|
|
int target_page_id = (int)page_count - 1;
|
|
int target_page = (int)cur_page;
|
|
|
|
if (direction == SCROLL_P_UP)
|
|
target_page = target_page+1 > target_page_id ? 0 : target_page+1;
|
|
else if (direction == SCROLL_P_DOWN)
|
|
target_page = target_page-1 < 0 ? target_page_id : target_page-1;
|
|
|
|
if (do_paint)
|
|
paintPage((uint8_t)target_page);
|
|
else
|
|
cur_page = (uint8_t)target_page;
|
|
|
|
OnAfterScrollPage();
|
|
}
|
|
|
|
bool CComponentsForm::clearSavedScreen()
|
|
{
|
|
std::lock_guard<std::mutex> g(cc_frm_mutex);
|
|
|
|
bool ret = false;
|
|
if (CCDraw::clearSavedScreen()){
|
|
for(size_t i=0; i<v_cc_items.size(); i++)
|
|
if (v_cc_items.at(i))
|
|
v_cc_items.at(i)->clearSavedScreen();
|
|
ret = true;
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
bool CComponentsForm::clearPaintCache()
|
|
{
|
|
std::lock_guard<std::mutex> g(cc_frm_mutex);
|
|
|
|
if (CCDraw::clearPaintCache()){
|
|
for(size_t i=0; i<v_cc_items.size(); i++)
|
|
if (v_cc_items.at(i))
|
|
v_cc_items.at(i)->clearPaintCache();
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
//clean old gradient buffer
|
|
bool CComponentsForm::clearFbGradientData()
|
|
{
|
|
std::lock_guard<std::mutex> g(cc_frm_mutex);
|
|
|
|
if (CCDraw::clearFbGradientData()){
|
|
for(size_t i=0; i<v_cc_items.size(); i++)
|
|
if (v_cc_items.at(i))
|
|
v_cc_items.at(i)->clearFbGradientData();
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
bool CComponentsForm::enableColBodyGradient(const int& enable_mode, const fb_pixel_t& sec_color, const int& direction)
|
|
{
|
|
std::lock_guard<std::mutex> g(cc_frm_mutex);
|
|
|
|
if (CCDraw::enableColBodyGradient(enable_mode, sec_color, direction)){
|
|
for (size_t i= 0; i< v_cc_items.size(); i++)
|
|
if (v_cc_items.at(i))
|
|
v_cc_items.at(i)->clearScreenBuffer();
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
int CComponentsForm::getUsedDY() const
|
|
{
|
|
int y_res = 0;
|
|
for (size_t i= 0; i< v_cc_items.size(); i++)
|
|
if (v_cc_items.at(i))
|
|
y_res = max(v_cc_items.at(i)->getYPos() + v_cc_items.at(i)->getHeight(), y_res);
|
|
|
|
return y_res;
|
|
}
|
|
|
|
int CComponentsForm::getUsedDX() const
|
|
{
|
|
int x_res = 0;
|
|
for (size_t i= 0; i< v_cc_items.size(); i++)
|
|
if (v_cc_items.at(i))
|
|
x_res = max(v_cc_items.at(i)->getXPos() + v_cc_items.at(i)->getWidth(), x_res);
|
|
|
|
return x_res;
|
|
}
|
|
|