Initial commit

This commit is contained in:
2023-11-26 20:13:49 +01:00
commit dc2dc5c58b
820 changed files with 258269 additions and 0 deletions

View File

@ -0,0 +1,29 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <platform/driver/touch/SDL2TouchController.hpp>
#include <platform/hal/simulator/sdl2/HALSDL2.hpp>
namespace touchgfx
{
void SDL2TouchController::init()
{
}
bool SDL2TouchController::sampleTouch(int32_t& x, int32_t& y)
{
return static_cast<HALSDL2*>(HAL::getInstance())->doSampleTouch(x, y); //lint !e1774
}
} // namespace touchgfx

View File

@ -0,0 +1,55 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <platform/hal/simulator/sdl2/HALSDL2.hpp>
namespace touchgfx
{
uint16_t HALSDL2::icon[] =
{
0x0000, 0x0AC1, 0x0AC9, 0x0ACD, 0x0ACE, 0x0ACE, 0x0ACE, 0x0ACE, 0x0ACE, 0x0ACE, 0x0ACE, 0x0ACE, 0x0ACE, 0x0ACE, 0x0ACE, 0x0ACE, 0x0ACE, 0x0ACE, 0x0ACE, 0x0ACE, 0x0ACE, 0x0ACE, 0x0ACE, 0x0ACE, 0x0ACE, 0x0ACE, 0x0ACE, 0x0ACE, 0x0ACD, 0x0AC9, 0x0AC1, 0x0000,
0x0AC1, 0x0ACE, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACE, 0x0AC1,
0x0AC9, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x09C9,
0x0ACD, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACD,
0x0ACE, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x6CDF, 0x7CEF, 0x7CEF, 0x7CEF, 0x5CDF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x5CDF, 0x7CEF, 0x7CEF, 0x7CEF, 0x6CDF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACD,
0x0ACE, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0xEFFF, 0xFFFF, 0xFFFF, 0xFFFF, 0xFFFF, 0x5BDF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x4BDF, 0xFFFF, 0xFFFF, 0xFFFF, 0xFFFF, 0xEFFF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x09CD,
0x09CE, 0x09CF, 0x09CF, 0x09CF, 0x0ACF, 0x0ACF, 0x09CF, 0x6CDF, 0xFFFF, 0xFFFF, 0xFFFF, 0xFFFF, 0xDFFF, 0x0ACF, 0x0ACF, 0x0ACF, 0x0ACF, 0x09CF, 0x09CF, 0xCEFF, 0xFFFF, 0xFFFF, 0xFFFF, 0xFFFF, 0x7CEF, 0x09CF, 0x09CF, 0x09CF, 0x09CF, 0x09CF, 0x09CF, 0x09CD,
0x09CE, 0x09CF, 0x09CF, 0x09CF, 0x09CF, 0x09CF, 0x09CF, 0x0ACF, 0xDEFF, 0xFFFF, 0xFFFF, 0xFFFF, 0xFFFF, 0x6CDF, 0x09CF, 0x09CF, 0x09CF, 0x09CF, 0x5BDF, 0xFFFF, 0xFFFF, 0xFFFF, 0xFFFF, 0xEFFF, 0x0ACF, 0x09CF, 0x09CF, 0x09CF, 0x09CF, 0x09CF, 0x09CF, 0x09CD,
0x09CE, 0x09CF, 0x09CF, 0x09CF, 0x09CF, 0x09CF, 0x09CF, 0x09CF, 0x5BDF, 0xFFFF, 0xFFFF, 0xFFFF, 0xFFFF, 0xDFFF, 0x09CF, 0x09CF, 0x09CF, 0x0ACF, 0xDFFF, 0xFFFF, 0xFFFF, 0xFFFF, 0xFFFF, 0x6CDF, 0x09CF, 0x09CF, 0x09CF, 0x09CF, 0x09CF, 0x09CF, 0x09CF, 0x09CD,
0x09CE, 0x09CF, 0x09CF, 0x09CF, 0x09CF, 0x09CF, 0x09CF, 0x09CF, 0x09CF, 0xCEFF, 0xFFFF, 0xFFFF, 0xFFFF, 0xBEEF, 0x09CF, 0x09CF, 0x09CF, 0x6CDF, 0xFFFF, 0xFFFF, 0xFFFF, 0xFFFF, 0xDEFF, 0x09CF, 0x09CF, 0x09CF, 0x09CF, 0x09CF, 0x09CF, 0x09CF, 0x09CF, 0x09CD,
0x0ACE, 0x09CF, 0x09CF, 0x0ACF, 0x09CF, 0x09CF, 0x09CF, 0x09CF, 0x09CF, 0x3BDF, 0xFFFF, 0xFFFF, 0xFFFF, 0x3ACF, 0x09CF, 0x09CF, 0x0ACF, 0xEFFF, 0xFFFF, 0xFFFF, 0xFFFF, 0xFFFF, 0x4BDF, 0x09CF, 0x09CF, 0x09CF, 0x09CF, 0x09CF, 0x09CF, 0x09CF, 0x09CF, 0x09CD,
0x09CE, 0x09CF, 0x09CF, 0x09CF, 0x09CF, 0x09CF, 0x09CF, 0x09CF, 0x09CF, 0x09CF, 0xBEEF, 0xFFFF, 0xADEF, 0x09CF, 0x09CF, 0x09CF, 0x7CDF, 0xFFFF, 0xFFFF, 0xFFFF, 0xFFFF, 0xCEFF, 0x09CF, 0x09CF, 0x09CF, 0x09CF, 0x09CF, 0x09CF, 0x09CF, 0x09CF, 0x09CF, 0x09CD,
0x09CE, 0x09CF, 0x09CF, 0x09CF, 0x09CF, 0x09CF, 0x09CF, 0x09CF, 0x09CF, 0x09CF, 0x2ACF, 0xFFFF, 0x2ACF, 0x09CF, 0x09CF, 0x1ACF, 0xEFFF, 0xFFFF, 0xFFFF, 0xFFFF, 0xFFFF, 0x3BCF, 0x09CF, 0x09BF, 0x09BF, 0x09BF, 0x09CF, 0x09CF, 0x09CF, 0x09CF, 0x09CF, 0x09CD,
0x09CE, 0x09BF, 0x09BF, 0x09CF, 0x09CF, 0x09CF, 0x09BF, 0x09CF, 0x09BF, 0x09BF, 0x09CF, 0x09CF, 0x09BF, 0x09BF, 0x09BF, 0x8CEF, 0xFFFF, 0xFFFF, 0xFFFF, 0xFFFF, 0xBDEF, 0x09CF, 0x09BF, 0x09BF, 0x09CF, 0x09CF, 0x09BF, 0x09CF, 0x09BF, 0x09BF, 0x09BF, 0x09CD,
0x09CE, 0x09BF, 0x09BF, 0x09BF, 0x09CF, 0x09CF, 0x09BF, 0x09BF, 0x09BF, 0x09BF, 0x09BF, 0x09BF, 0x09BF, 0x09BF, 0x1ACF, 0xFFFF, 0xFFFF, 0xFFFF, 0xFFFF, 0xFFFF, 0x2ACF, 0x09BF, 0x09BF, 0x09BF, 0x09BF, 0x09BF, 0x09BF, 0x09BF, 0x09BF, 0x09CF, 0x09CF, 0x09BD,
0x09BE, 0x09BF, 0x09CF, 0x09BF, 0x09BF, 0x09BF, 0x09BF, 0x09BF, 0x09BF, 0x09BF, 0x09BF, 0x09BF, 0x09BF, 0x09BF, 0x9DEF, 0xFFFF, 0xFFFF, 0xFFFF, 0xFFFF, 0xADEF, 0x09BF, 0x09BF, 0x09BF, 0x09BF, 0x09BF, 0x09BF, 0x09BF, 0x09BF, 0x09BF, 0x09BF, 0x09BF, 0x09BD,
0x09BE, 0x09CF, 0x09CF, 0x09CF, 0x09BF, 0x09BF, 0x09BF, 0x09BF, 0x09BF, 0x09BF, 0x09BF, 0x09BF, 0x09BF, 0x09BF, 0x8CDF, 0xFFFF, 0xFFFF, 0xFFFF, 0xFFFF, 0xBEEF, 0x09BF, 0x09BF, 0x09BF, 0x09BF, 0x09BF, 0x09BF, 0x09BF, 0x09BF, 0x09BF, 0x09BF, 0x09BF, 0x09BD,
0x09BE, 0x09BF, 0x09BF, 0x09BF, 0x09CF, 0x09BF, 0x09BF, 0x09BF, 0x09BF, 0x09BF, 0x09BF, 0x09BF, 0x09BF, 0x09BF, 0x19BF, 0xEFFF, 0xFFFF, 0xFFFF, 0xFFFF, 0xFFFF, 0x4BCF, 0x09BF, 0x09BF, 0x09BF, 0x09BF, 0x09BF, 0x09BF, 0x09BF, 0x09BF, 0x09BF, 0x09BF, 0x09BD,
0x09BE, 0x09BF, 0x09BF, 0x09BF, 0x09BF, 0x09BF, 0x09BF, 0x09BF, 0x09BF, 0x09BF, 0x09BF, 0x3ACF, 0x09BF, 0x09BF, 0x09BF, 0x6CDF, 0xFFFF, 0xFFFF, 0xFFFF, 0xFFFF, 0xCEFF, 0x09BF, 0x09BF, 0x09BF, 0x09BF, 0x09BF, 0x09BF, 0x09BF, 0x09BF, 0x09BF, 0x09BF, 0x09BD,
0x09BE, 0x09BF, 0x09BF, 0x09BF, 0x09BF, 0x09BF, 0x09BF, 0x09BF, 0x09BF, 0x09BF, 0x4BCF, 0xFFFF, 0x3ACF, 0x09BF, 0x09BF, 0x09BF, 0xDEFF, 0xFFFF, 0xFFFF, 0xFFFF, 0xFFFF, 0x6BCF, 0x09BF, 0x09BF, 0x09BF, 0x09BF, 0x09BF, 0x09BF, 0x09BF, 0x09BF, 0x09BF, 0x09BD,
0x09BE, 0x09BF, 0x09BF, 0x09BF, 0x09BF, 0x08BF, 0x09BF, 0x09BF, 0x09BF, 0x09BF, 0xDEFF, 0xFFFF, 0xCEEF, 0x09BF, 0x09BF, 0x09BF, 0x4BCF, 0xFFFF, 0xFFFF, 0xFFFF, 0xFFFF, 0xEFFF, 0x09BF, 0x08BF, 0x09BF, 0x09BF, 0x09BF, 0x09BF, 0x09BF, 0x09BF, 0x09BF, 0x09BD,
0x09BE, 0x09BF, 0x09BF, 0x09BF, 0x09BF, 0x09BF, 0x08BF, 0x09BF, 0x09BF, 0x6BDF, 0xFFFF, 0xFFFF, 0xFFFF, 0x5BCF, 0x09BF, 0x09BF, 0x09BF, 0xBEEF, 0xFFFF, 0xFFFF, 0xFFFF, 0xFFFF, 0x7CDF, 0x09BF, 0x09BF, 0x08BF, 0x09BF, 0x09BF, 0x09BF, 0x09BF, 0x09BF, 0x09BD,
0x09BE, 0x09BF, 0x09BF, 0x09BF, 0x09BF, 0x08BF, 0x08BF, 0x09BF, 0x19BF, 0xEFFF, 0xFFFF, 0xFFFF, 0xFFFF, 0xDEFF, 0x09BF, 0x09BF, 0x09BF, 0x3ACF, 0xFFFF, 0xFFFF, 0xFFFF, 0xFFFF, 0xFFFF, 0x19BF, 0x08BF, 0x08BF, 0x08BF, 0x09BF, 0x08BF, 0x09BF, 0x09BF, 0x09BD,
0x08BE, 0x08BF, 0x08BF, 0x08BF, 0x08BF, 0x08BF, 0x09BF, 0x09BF, 0x8CDF, 0xFFFF, 0xFFFF, 0xFFFF, 0xFFFF, 0xADEF, 0x08BF, 0x08BF, 0x09BF, 0x08BF, 0x9DEF, 0xFFFF, 0xFFFF, 0xFFFF, 0xFFFF, 0x9DEF, 0x08AF, 0x08AF, 0x08BF, 0x08BF, 0x08BF, 0x08BF, 0x08AF, 0x08AD,
0x08AE, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x29BF, 0xFFFF, 0xFFFF, 0xFFFF, 0xFFFF, 0xFFFF, 0x29BF, 0x08AF, 0x09BF, 0x08AF, 0x08AF, 0x19BF, 0xFFFF, 0xFFFF, 0xFFFF, 0xFFFF, 0xFFFF, 0x3ABF, 0x08BF, 0x08AF, 0x08AF, 0x08BF, 0x08AF, 0x08AF, 0x09BD,
0x09BE, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0xADEF, 0xFFFF, 0xFFFF, 0xFFFF, 0xFFFF, 0x9CDF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x8CDF, 0xFFFF, 0xFFFF, 0xFFFF, 0xFFFF, 0xBEEF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x08AD,
0x09BE, 0x09BF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x29BF, 0xFFFF, 0xFFFF, 0xFFFF, 0xFFFF, 0xEFFF, 0x19BF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x08BF, 0x08AF, 0xEFFF, 0xFFFF, 0xFFFF, 0xFFFF, 0xFFFF, 0x3ABF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x08AD,
0x09BE, 0x09BF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x29BF, 0x39BF, 0x39BF, 0x39BF, 0x19BF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x08BF, 0x18AF, 0x3ABF, 0x3ABF, 0x39BF, 0x2ABF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x08AD,
0x09BD, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x08AD,
0x08A9, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x08A9,
0x08A1, 0x08AE, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x089F, 0x089F, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x08AF, 0x079F, 0x079E, 0x0791,
0x0000, 0x07A1, 0x08A9, 0x08AD, 0x09BD, 0x08AD, 0x08AD, 0x079D, 0x079D, 0x079D, 0x079D, 0x08AD, 0x08AD, 0x08AD, 0x08AD, 0x089D, 0x079D, 0x079D, 0x08AD, 0x08AD, 0x08AD, 0x08AD, 0x08AD, 0x079D, 0x079D, 0x079D, 0x08AD, 0x08AD, 0x08AD, 0x0799, 0x0791, 0x0000
};
} // namespace touchgfx

View File

@ -0,0 +1,50 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <touchgfx/hal/OSWrappers.hpp>
#include <SDL2/SDL.h>
namespace touchgfx
{
static SDL_mutex* s_FrameBufferLock;
void OSWrappers::initialize()
{
// Setup framebuffer lock
s_FrameBufferLock = SDL_CreateMutex();
}
void OSWrappers::takeFrameBufferSemaphore()
{
SDL_LockMutex(s_FrameBufferLock);
}
void OSWrappers::giveFrameBufferSemaphore()
{
SDL_UnlockMutex(s_FrameBufferLock);
}
void OSWrappers::waitForVSync()
{
}
void OSWrappers::tryTakeFrameBufferSemaphore()
{
}
void OSWrappers::giveFrameBufferSemaphoreFromISR()
{
}
} // namespace touchgfx

View File

@ -0,0 +1,112 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <touchgfx/containers/CacheableContainer.hpp>
#include <touchgfx/hal/HAL.hpp>
namespace touchgfx
{
CacheableContainer::CacheableContainer()
: Container(),
cachedBitmapId(BITMAP_INVALID),
cachedImage(),
isCachedMode(false),
childWasInvalidated(false)
{
}
void CacheableContainer::setCacheBitmap(BitmapId bitmapId)
{
Bitmap bitmap(bitmapId);
cachedBitmapId = BITMAP_INVALID;
/* Accept only dynamic bitmaps */
if (!Bitmap::dynamicBitmapGetAddress(bitmapId))
{
return;
}
/* Retrieve the auxiliary LCD implementation attached to HA */
LCD* auxLCD = HAL::getInstance()->getAuxiliaryLCD();
/* Dynamic bitmap and framebuffer should be of the same format */
if (bitmap.getFormat() == HAL::lcd().framebufferFormat()
|| (auxLCD && bitmap.getFormat() == auxLCD->framebufferFormat()))
{
cachedBitmapId = bitmapId;
cachedImage.setBitmap(Bitmap(cachedBitmapId));
}
}
BitmapId CacheableContainer::getCacheBitmap() const
{
return cachedBitmapId;
}
void CacheableContainer::updateCache()
{
updateCache(Rect());
}
void CacheableContainer::updateCache(const Rect& invalidatedArea)
{
if (isCachedMode && (cachedBitmapId != BITMAP_INVALID))
{
/* will call Container::draw(invalidatedArea) to render all widgets into the dynamic bitmap */
HAL::getInstance()->drawDrawableInDynamicBitmap(*this, cachedBitmapId, invalidatedArea);
childWasInvalidated = false;
}
}
void CacheableContainer::enableCachedMode(bool enable)
{
isCachedMode = enable;
}
void CacheableContainer::invalidateRect(Rect& invalidatedArea) const
{
Container::invalidateRect(invalidatedArea);
if (isCachedMode && (cachedBitmapId != BITMAP_INVALID))
{
const_cast<CacheableContainer*>(this)->childWasInvalidated = true;
}
}
bool CacheableContainer::setSolidRect(const Rect& rect)
{
return Bitmap::dynamicBitmapSetSolidRect(cachedBitmapId, rect);
}
bool CacheableContainer::isChildInvalidated() const
{
return childWasInvalidated;
}
void CacheableContainer::setupDrawChain(const Rect& invalidatedArea, Drawable** nextPreviousElement)
{
if (isCachedMode && (cachedBitmapId != BITMAP_INVALID) && isVisible())
{
Rect r = getAbsoluteRect();
cachedImage.setPosition(r.x, r.y, r.width, r.height);
cachedImage.setupDrawChain(invalidatedArea, nextPreviousElement);
}
else
{
Container::setupDrawChain(invalidatedArea, nextPreviousElement);
}
}
} // namespace touchgfx

View File

@ -0,0 +1,276 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <touchgfx/containers/Container.hpp>
#include <touchgfx/Screen.hpp>
namespace touchgfx
{
bool Container::contains(const Drawable& d)
{
bool found = false;
Drawable* head = firstChild;
while (head && !found)
{
found = (head == &d);
head = head->nextSibling;
}
return found;
}
void Container::add(Drawable& d)
{
assert(&d != this && "Cannot add Drawable to self");
assert(d.parent == 0 && "Cannot add Drawable multiple times");
// Initialize d to have this as parent and no sibling.
d.parent = this;
d.nextSibling = 0;
// Check if d is the first child to be added (container is empty)
if (!firstChild)
{
firstChild = &d;
}
else
{
Drawable* head = firstChild;
// Skip to end of currently added children
while (head->nextSibling)
{
assert(head != &d && "Cannot add Drawable multiple times");
head = head->nextSibling;
}
assert(head != &d && "Cannot add Drawable multiple times");
// Make last child now point to d.
head->nextSibling = &d;
}
}
void Container::remove(Drawable& d)
{
if (!firstChild)
{
// No children
return;
}
if (&d == firstChild)
{
// d is first child.
d.parent = 0;
if (!d.nextSibling)
{
// d was only child, so now this container is empty
firstChild = 0;
}
else
{
firstChild = d.nextSibling;
d.nextSibling = 0;
}
return;
}
Drawable* tmp = firstChild;
while (tmp)
{
if (tmp->nextSibling == &d)
{
tmp->nextSibling = d.nextSibling;
d.parent = 0;
d.nextSibling = 0;
return;
}
else
{
tmp = tmp->nextSibling;
}
}
}
void Container::removeAll()
{
while (firstChild)
{
Drawable* d = firstChild;
firstChild = firstChild->nextSibling;
d->parent = 0;
d->nextSibling = 0;
}
}
void Container::unlink()
{
firstChild = 0;
}
void Container::draw(const Rect& invalidatedArea) const
{
// The draw function of Container is not normally used. Containers do not per default
// appear in the draw chain, since they are normally invisible themselves. However,
// if someone decides to call draw on a container, at least do something useful (draw children).
if (!isVisible() || !firstChild)
{
// Nothing to draw
return;
}
Rect tmp = invalidatedArea;
Drawable* d = firstChild;
while (d)
{
if (d->isVisible())
{
Rect drawableRegion = tmp & d->getRect();
if (!drawableRegion.isEmpty())
{
// This child has a non-empty intersection with the invalidated area.
// Convert region to the Drawable's coordinate system and draw.
drawableRegion.x -= d->getX();
drawableRegion.y -= d->getY();
d->draw(drawableRegion);
}
}
d = d->nextSibling;
}
}
void Container::getLastChild(int16_t x, int16_t y, Drawable** last)
{
// This function is used to obtain the drawable that should receive a click/drag/gesture event.
// Find the last child (ie. the last child that was added, ie. the "front-most" drawable) covering
// the specified coords.
if (isTouchable())
{
// If the container itself is touchable, result so far is "this". Might be overridden by a child.
*last = this;
}
Drawable* d = firstChild;
while (d)
{
// Iterate over children.
if (d->isVisible() && d->getRect().intersect(x, y))
{
int16_t xadj = x - d->getX();
int16_t yadj = y - d->getY();
d->getLastChild(xadj, yadj, last);
}
d = d->nextSibling;
}
}
Rect Container::getSolidRect() const
{
return Rect(0, 0, 0, 0);
}
Rect Container::getContainedArea() const
{
Drawable* d = firstChild;
Rect contained(0, 0, 0, 0);
while (d)
{
contained.expandToFit(d->getRect());
d = d->nextSibling;
}
return contained;
}
void Container::moveChildrenRelative(int16_t deltaX, int16_t deltaY)
{
Drawable* d = firstChild;
while (d)
{
d->moveRelative(deltaX, deltaY);
d = d->nextSibling;
}
}
void Container::forEachChild(GenericCallback<Drawable&>* function)
{
Drawable* d = firstChild;
while (d)
{
function->execute(*d);
d = d->nextSibling;
}
}
void Container::insert(Drawable* previous, Drawable& d)
{
if (!firstChild)
{
// Insert as only element
add(d);
return;
}
else if (!previous)
{
// Insert as head element
d.nextSibling = firstChild;
firstChild = &d;
d.parent = this;
}
else
{
Drawable* tmp = firstChild;
while (tmp)
{
if (tmp == previous)
{
d.nextSibling = tmp->nextSibling;
tmp->nextSibling = &d;
d.parent = this;
return;
}
tmp = tmp->nextSibling;
}
}
}
void Container::setupDrawChain(const Rect& invalidatedArea, Drawable** nextPreviousElement)
{
// This function adds the children of this container to the list of drawables to draw.
if (!isVisible())
{
// If this container itself is not visible, do not add anyone to draw chain.
return;
}
if (!firstChild)
{
// If this container is empty, do not add anyone.
return;
}
Drawable* d = firstChild;
while (d)
{
if (d->isVisible())
{
// Only drawables intersecting with the specified invalidated area will be added.
Rect drawableRegion = invalidatedArea & d->getRect();
if (!drawableRegion.isEmpty())
{
drawableRegion.x -= d->getX();
drawableRegion.y -= d->getY();
d->setupDrawChain(drawableRegion, nextPreviousElement);
}
}
d = d->nextSibling;
}
}
} // namespace touchgfx

View File

@ -0,0 +1,248 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <touchgfx/containers/ListLayout.hpp>
namespace touchgfx
{
class AdjustElements
{
public:
AdjustElements(Drawable* d, Direction dir)
: insertedCoord(0),
newElementPassed(false),
newElement(d),
direction(dir)
{
}
void handleInsert(Drawable& d)
{
if (!newElementPassed)
{
if (newElement == &d)
{
newElementPassed = true;
}
else
{
if (direction == SOUTH)
{
insertedCoord = d.getY() + d.getHeight();
}
else if (direction == EAST)
{
insertedCoord = d.getX() + d.getWidth();
}
}
}
else
{
if (direction == SOUTH)
{
d.setY(d.getY() + newElement->getHeight());
}
else if (direction == EAST)
{
d.setX(d.getX() + newElement->getWidth());
}
}
}
void handleRemove(Drawable& d)
{
if (!newElementPassed)
{
if (newElement == &d)
{
newElementPassed = true;
}
}
else
{
if (direction == SOUTH)
{
d.setY(d.getY() - newElement->getHeight());
}
else if (direction == EAST)
{
d.setX(d.getX() - newElement->getWidth());
}
}
if (newElement != &d)
{
if (direction == SOUTH)
{
if (d.getWidth() > insertedCoord)
{
insertedCoord = d.getWidth();
}
}
else if (direction == EAST)
{
if (d.getHeight() > insertedCoord)
{
insertedCoord = d.getHeight();
}
}
}
}
int16_t insertedCoord;
bool newElementPassed;
private:
Drawable* newElement;
Direction direction;
}; //lint !e1712
void ListLayout::internalAddElementAt(Drawable& d, int16_t coord)
{
switch (direction)
{
case SOUTH:
if (rect.width < d.getWidth())
{
rect.width = d.getWidth();
}
rect.height += d.getHeight();
d.setXY(0, coord);
offset += d.getHeight();
break;
case EAST:
if (rect.height < d.getHeight())
{
rect.height = d.getHeight();
}
rect.width += d.getWidth();
d.setXY(coord, 0);
offset += d.getWidth();
break;
case NORTH:
case WEST:
default:
break;
}
}
void ListLayout::internalAddElement(Drawable& d)
{
internalAddElementAt(d, offset);
}
void ListLayout::internalRemoveElement(Drawable& d, int16_t coord)
{
switch (direction)
{
case SOUTH:
if (rect.width > coord)
{
rect.width = coord;
}
rect.height -= d.getHeight();
d.setX(0);
d.setY(0);
offset -= d.getHeight();
break;
case EAST:
if (rect.height > coord)
{
rect.height = coord;
}
rect.width -= d.getWidth();
d.setX(0);
d.setY(0);
offset -= d.getWidth();
break;
case NORTH:
case WEST:
default:
break;
}
}
void ListLayout::setDirection(const Direction d)
{
assert((d == SOUTH || d == EAST) && "Chosen direction not supported");
if (direction != d)
{
direction = d;
offset = 0;
setWidthHeight(0, 0);
Callback<ListLayout, Drawable&> function(this, &ListLayout::internalAddElement);
forEachChild(&function);
if (parent)
{
parent->childGeometryChanged();
}
}
}
void ListLayout::add(Drawable& d)
{
internalAddElement(d);
Container::add(d);
if (parent)
{
parent->childGeometryChanged();
}
}
void ListLayout::removeAll()
{
offset = 0;
setWidthHeight(0, 0);
Container::removeAll();
if (parent)
{
parent->childGeometryChanged();
}
}
void ListLayout::insert(Drawable* previous, Drawable& d)
{
if (!firstChild)
{
// List is empty, just add the new entry
add(d);
return;
}
Container::insert(previous, d);
AdjustElements tmp(&d, direction);
Callback<AdjustElements, Drawable&> function(&tmp, &AdjustElements::handleInsert);
forEachChild(&function);
internalAddElementAt(d, tmp.insertedCoord);
if (parent)
{
parent->childGeometryChanged();
}
}
void ListLayout::remove(Drawable& d)
{
AdjustElements tmp(&d, direction);
Callback<AdjustElements, Drawable&> function(&tmp, &AdjustElements::handleRemove);
forEachChild(&function);
if (tmp.newElementPassed)
{
internalRemoveElement(d, tmp.insertedCoord);
Container::remove(d);
}
if (parent)
{
parent->childGeometryChanged();
}
}
} // namespace touchgfx

View File

@ -0,0 +1,107 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <touchgfx/containers/ModalWindow.hpp>
#include <touchgfx/Color.hpp>
namespace touchgfx
{
ModalWindow::ModalWindow()
: Container()
{
ModalWindow::setWidthHeight(HAL::DISPLAY_WIDTH, HAL::DISPLAY_HEIGHT);
int defaultShadeAlpha = 96;
colortype defaultShadeColor = Color::getColorFrom24BitRGB(0x0, 0x0, 0x0);
backgroundShade.setPosition(0, 0, getWidth(), getHeight());
backgroundShade.setColor(defaultShadeColor);
backgroundShade.setTouchable(true);
ModalWindow::setShadeAlpha(defaultShadeAlpha);
Container::add(backgroundShade);
Container::add(windowContainer);
windowContainer.add(windowBackground);
}
void ModalWindow::setBackground(const BitmapId& bmpId)
{
windowBackground.setBitmap(Bitmap(bmpId));
windowBackground.setXY(0, 0);
windowContainer.setPosition((getWidth() - windowBackground.getWidth()) / 2, (getHeight() - windowBackground.getHeight()) / 2, windowBackground.getWidth(), windowBackground.getHeight());
invalidate();
}
void ModalWindow::setBackground(const BitmapId& bmpId, int16_t backgroundX, int16_t backgroundY)
{
setBackground(bmpId);
windowContainer.setXY(backgroundX, backgroundY);
}
uint16_t ModalWindow::getBackgroundWidth() const
{
return windowBackground.getWidth();
}
uint16_t ModalWindow::getBackgroundHeight() const
{
return windowBackground.getHeight();
}
void ModalWindow::add(Drawable& d)
{
windowContainer.add(d);
}
void ModalWindow::remove(Drawable& d)
{
windowContainer.remove(d);
}
void ModalWindow::setShadeAlpha(uint8_t alpha)
{
backgroundShade.setAlpha(alpha);
backgroundShade.invalidate();
}
uint8_t ModalWindow::getShadeAlpha() const
{
return backgroundShade.getAlpha();
}
void ModalWindow::setShadeColor(colortype color)
{
backgroundShade.setColor(color);
backgroundShade.invalidate();
}
colortype ModalWindow::getShadeColor() const
{
return backgroundShade.getColor();
}
void ModalWindow::show()
{
setVisible(true);
invalidate();
}
void ModalWindow::hide()
{
setVisible(false);
invalidate();
}
} // namespace touchgfx

View File

@ -0,0 +1,730 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <touchgfx/containers/ScrollableContainer.hpp>
#include <touchgfx/EasingEquations.hpp>
#include <touchgfx/Color.hpp>
namespace touchgfx
{
ScrollableContainer::ScrollableContainer()
: Container(),
scrollbarPadding(0),
scrollbarWidth(2),
scrollbarAlpha(120),
scrollbarColor(Color::getColorFrom24BitRGB(0xFF, 0xFF, 0xFF)),
accelDirection(GestureEvent::SWIPE_HORIZONTAL),
xSlider(0, 0, scrollbarColor, scrollbarAlpha),
ySlider(0, 0, scrollbarColor, scrollbarAlpha),
pressedDrawable(0),
lastDraggableChild(0),
scrolledXDistance(0),
scrolledYDistance(0),
scrollThreshold(5),
pressedX(0),
pressedY(0),
isPressed(false),
isScrolling(false),
scrollableX(true),
scrollableY(true),
scrollbarsVisible(true),
scrollbarsPermanentlyVisible(false),
scrollDuration(0),
beginningValue(0),
targetValue(0),
animationCounter(0),
animate(false),
fingerAdjustmentX(0),
fingerAdjustmentY(0),
hasIssuedCancelEvent(false),
scrollDurationSpeedup(7),
scrollDurationSlowdown(1)
{
xSlider.setVisible(false);
ySlider.setVisible(false);
maxVelocity = SCROLLBAR_MAX_VELOCITY;
setTouchable(true);
}
void ScrollableContainer::handleClickEvent(const ClickEvent& evt)
{
if (evt.getType() == ClickEvent::PRESSED)
{
isPressed = true;
if (animate)
{
// Stop scroll animation
animate = false;
animationCounter = 0;
Application::getInstance()->unregisterTimerWidget(this);
}
const int fingerSize = HAL::getInstance()->getFingerSize();
fingerAdjustmentX = 0;
fingerAdjustmentY = 0;
const int minimumDistance = 3;
if ((fingerSize - 1) >= minimumDistance)
{
pressedDrawable = 0;
const int maximumSquares = 3;
const int numberOfSquares = MIN(maximumSquares, (fingerSize - 1) / minimumDistance);
uint32_t bestDistance = 0xFFFFFFFF;
Drawable* last = 0;
Rect me(0, 0, getWidth(), getHeight());
Rect meAbs = me;
translateRectToAbsolute(meAbs);
for (int squareNumber = 1; squareNumber <= numberOfSquares; squareNumber++)
{
int distance = ((squareNumber * fingerSize) / numberOfSquares);
const int samplePoints[10][2] = { { 0, 0 }, { -1, -1 }, { 0, -1 }, { 1, -1 }, { -1, 0 }, { 0, 0 }, { 1, 0 }, { -1, 1 }, { 0, 1 }, { 1, 1 } };
for (int sampleIndex = squareNumber - 1; sampleIndex < 10; sampleIndex += 2)
{
Drawable* d = 0;
int16_t deltaX = samplePoints[sampleIndex][0] * distance;
int16_t deltaY = samplePoints[sampleIndex][1] * distance;
if (me.intersect(evt.getX() + deltaX, evt.getY() + deltaY))
{
Container::getLastChild(evt.getX() + deltaX, evt.getY() + deltaY, &d);
if (d && d != last && d != this)
{
Rect absRect = d->getAbsoluteRect();
int x = meAbs.x + evt.getX() - (absRect.x + (absRect.width / 2));
int y = meAbs.y + evt.getY() - (absRect.y + (absRect.height / 2));
uint32_t dist = x * x + y * y;
if (dist < bestDistance)
{
last = d;
bestDistance = dist;
pressedDrawable = d;
fingerAdjustmentX = deltaX;
fingerAdjustmentY = deltaY;
}
}
}
}
}
}
else
{
Container::getLastChild(evt.getX(), evt.getY(), &pressedDrawable);
}
if (pressedDrawable == this)
{
pressedDrawable = 0;
}
if (pressedDrawable)
{
hasIssuedCancelEvent = false;
pressedX = evt.getX();
pressedY = evt.getY();
Rect r = pressedDrawable->getAbsoluteRect();
ClickEvent relative(evt.getType(), evt.getX() + rect.x + fingerAdjustmentX - r.x, evt.getY() + rect.y + fingerAdjustmentY - r.y);
pressedDrawable->handleClickEvent(relative);
lastDraggableChild = pressedDrawable;
}
}
else if (evt.getType() == ClickEvent::CANCEL)
{
return;
}
else // if (evt.getType() == ClickEvent::RELEASED)
{
if (pressedDrawable)
{
Rect r = pressedDrawable->getAbsoluteRect();
ClickEvent relative(evt.getType(), evt.getX() + rect.x + fingerAdjustmentX - r.x, evt.getY() + rect.y + fingerAdjustmentY - r.y);
pressedDrawable->handleClickEvent(relative);
}
pressedDrawable = 0;
lastDraggableChild = 0;
isPressed = false;
}
isScrolling = false;
// Redraw scrollbars.
xSlider.setVisible((isPressed && scrollbarsVisible) || scrollbarsPermanentlyVisible);
ySlider.setVisible((isPressed && scrollbarsVisible) || scrollbarsPermanentlyVisible);
invalidateScrollbars();
}
void ScrollableContainer::handleDragEvent(const DragEvent& evt)
{
DragEvent actualDrag = evt;
bool acceptInitialScroll = false;
bool canScrollX = false;
bool canScrollY = false;
isScrollableXY(canScrollX, canScrollY);
if ((pressedDrawable != 0) && (pressedDrawable != this))
{
// Also send this drag event to the appropriate child widget
Rect r = pressedDrawable->getAbsoluteRect();
int16_t oldX = evt.getOldX() + rect.x + fingerAdjustmentX - r.x;
int16_t oldY = evt.getOldY() + rect.y + fingerAdjustmentY - r.y;
int16_t newX = canScrollX ? oldX : evt.getNewX() + rect.x + fingerAdjustmentX - r.x;
int16_t newY = canScrollY ? oldY : evt.getNewY() + rect.y + fingerAdjustmentY - r.y;
// but only in the direction(s) where the scrollable container itself
// cannot scroll.
if ((!canScrollX && newX != oldX) || (!canScrollY && newY != oldY))
{
DragEvent relative(DragEvent::DRAGGED, oldX, oldY, newX, newY);
pressedDrawable->handleDragEvent(relative);
}
}
// If we are not currently scrolling, the drag event delta must
// be larger than the threshold value, otherwise the event is ignored.
if (!isScrolling)
{
// Only consider the delta in directions that are actually scrollable.
// Note: Do not use the delta from received evt since that only reflects
// change since last drag. What we want to check here is if the total
// delta from the point of click has now exceeded the threshold.
actualDrag = DragEvent(DragEvent::DRAGGED, pressedX + fingerAdjustmentX, pressedY + fingerAdjustmentY, evt.getNewX() + fingerAdjustmentX, evt.getNewY() + fingerAdjustmentY);
if (canScrollX)
{
// Can scroll in X.
if (abs(actualDrag.getDeltaX()) > scrollThreshold)
{
acceptInitialScroll = true;
}
}
if (canScrollY)
{
// Can scroll in Y.
if (abs(actualDrag.getDeltaY()) > scrollThreshold)
{
acceptInitialScroll = true;
}
}
if (acceptInitialScroll)
{
isScrolling = true;
}
else
{
// Discard this drag event. However, if the new coordinates no longer matches the drawable which received the PRESSED click event
// issue a CANCEL event to that drawable.
if (pressedDrawable && !hasIssuedCancelEvent)
{
Drawable* child = 0;
Container::getLastChild(evt.getNewX() + fingerAdjustmentX, evt.getNewY() + fingerAdjustmentY, &child);
if (pressedDrawable != child)
{
ClickEvent ce(ClickEvent::CANCEL, 0, 0);
pressedDrawable->handleClickEvent(ce);
hasIssuedCancelEvent = true;
}
}
return;
}
}
// Send cancel events to child in focus
if (pressedDrawable && !hasIssuedCancelEvent)
{
ClickEvent ce(ClickEvent::CANCEL, 0, 0);
pressedDrawable->handleClickEvent(ce);
hasIssuedCancelEvent = true;
}
int16_t deltaX = 0;
int16_t deltaY = 0;
if (scrollableX)
{
if (acceptInitialScroll)
{
// Initial drag which is past threshold, only scroll one pixel in initial event.
if (actualDrag.getDeltaX() > 0)
{
deltaX = 1;
}
else if (actualDrag.getDeltaX() < 0)
{
deltaX = -1;
}
}
else
{
// Scroll entire delta
deltaX = actualDrag.getDeltaX();
}
}
else
{
// Not scrollable
deltaX = 0;
}
if (scrollableY)
{
if (acceptInitialScroll)
{
// Initial drag which is past threshold, only scroll one pixel in initial event.
if (actualDrag.getDeltaY() > 0)
{
deltaY = 1;
}
else if (actualDrag.getDeltaY() < 0)
{
deltaY = -1;
}
}
else
{
// Scroll entire delta
deltaY = actualDrag.getDeltaY();
}
}
else
{
// Not scrollable
deltaY = 0;
}
doScroll(deltaX, deltaY);
}
void ScrollableContainer::handleGestureEvent(const GestureEvent& evt)
{
bool canScrollX = false;
bool canScrollY = false;
isScrollableXY(canScrollX, canScrollY);
if ((canScrollX && (evt.getType() == GestureEvent::SWIPE_HORIZONTAL)) ||
(canScrollY && (evt.getType() == GestureEvent::SWIPE_VERTICAL)))
{
int16_t velocityAbsolute = abs(evt.getVelocity());
// Ignore gestures with velocity lower than threshold
if (velocityAbsolute < scrollThreshold)
{
return;
}
// Force velocity within limits
velocityAbsolute = MIN(velocityAbsolute, maxVelocity);
velocityAbsolute = MAX(velocityAbsolute, SCROLLBAR_MIN_VELOCITY);
// Try to set some reasonable values for how long the resulting scroll should be, and how many ticks is should take
scrollDuration = velocityAbsolute * scrollDurationSpeedup / scrollDurationSlowdown;
targetValue = ((evt.getVelocity() > 0) ? 1 : -1) * (velocityAbsolute - 4) * 72;
scrollDuration = MIN(scrollDuration, abs(targetValue));
// Get ready to animate scroll: Initialize values
beginningValue = (evt.getType() == GestureEvent::SWIPE_VERTICAL) ? getContainedArea().y : getContainedArea().x;
animate = true;
Application::getInstance()->registerTimerWidget(this);
accelDirection = evt.getType();
if (pressedDrawable && !hasIssuedCancelEvent)
{
ClickEvent ce(ClickEvent::CANCEL, 0, 0);
pressedDrawable->handleClickEvent(ce);
hasIssuedCancelEvent = true;
}
}
}
Rect ScrollableContainer::getXScrollbar() const
{
Rect res(0, 0, 0, 0);
if (scrollableX)
{
Rect contained = getContainedArea();
const int scrollSpace = (scrollableY && (contained.height > rect.height)) ? (2 * scrollbarPadding + scrollbarWidth + SCROLLBAR_LINE) : 0;
if (contained.width > rect.width)
{
int leftPadding = (-1 * contained.x * rect.width) / contained.width;
int rightPadding = ((contained.right() - rect.width) * rect.width) / contained.width;
const int startWidth = rect.width - 2 * scrollbarPadding - 2 * SCROLLBAR_LINE - scrollSpace;
int width = startWidth;
width -= (leftPadding + rightPadding);
if (width < scrollbarWidth * 2)
{
// If the contained area is very large, the scrollbar width may become zero or even negative.
int diff = scrollbarWidth * 2 - width;
width = scrollbarWidth * 2; // Force scrollbar width to a minimum
// Distribute the deviation error based on current scrollbar X position (the amount subtracted from scrollbar xpos increases gradually).
leftPadding -= (diff * leftPadding) / startWidth;
}
res = Rect(leftPadding + scrollbarPadding + SCROLLBAR_LINE, rect.height - scrollbarWidth - scrollbarPadding - SCROLLBAR_LINE, width, scrollbarWidth);
}
}
return res;
}
Rect ScrollableContainer::getYScrollbar() const
{
Rect res(0, 0, 0, 0);
if (scrollableY)
{
Rect contained = getContainedArea();
const int scrollSpace = (scrollableX && (contained.width > rect.width)) ? (2 * scrollbarPadding + scrollbarWidth + SCROLLBAR_LINE) : 0;
if (contained.height > rect.height)
{
int topPadding = (-1 * contained.y * rect.height) / contained.height;
int bottomPadding = ((contained.bottom() - rect.height) * rect.height) / contained.height;
const int startHeight = rect.height - 2 * scrollbarPadding - 2 * SCROLLBAR_LINE - scrollSpace;
int height = startHeight;
height -= (topPadding + bottomPadding);
if (height < scrollbarWidth * 2)
{
// If the contained area is very large, the scrollbar height may become zero or even negative.
int diff = scrollbarWidth * 2 - height;
height = scrollbarWidth * 2; // Force scrollbar height to a minimum
// Distribute the deviation error based on current scrollbar Y position (the amount subtracted from scrollbar ypos increases gradually).
topPadding -= (diff * topPadding) / startHeight;
}
res = Rect(rect.width - scrollbarWidth - scrollbarPadding - 2 * SCROLLBAR_LINE, topPadding + scrollbarPadding + SCROLLBAR_LINE, scrollbarWidth, height);
}
}
return res;
}
Rect ScrollableContainer::getXBorder(const Rect& xBar, const Rect& yBar) const
{
Rect border(0, 0, 0, 0);
if (!xBar.isEmpty())
{
const int scrollSpace = (!yBar.isEmpty()) ? (2 * scrollbarPadding + scrollbarWidth + SCROLLBAR_LINE) : 0;
border = Rect(scrollbarPadding, xBar.y - SCROLLBAR_LINE, rect.width - 2 * scrollbarPadding - scrollSpace, scrollbarWidth + 2 * SCROLLBAR_LINE);
}
return border;
}
Rect ScrollableContainer::getYBorder(const Rect& xBar, const Rect& yBar) const
{
Rect border(0, 0, 0, 0);
if (!yBar.isEmpty())
{
const int scrollSpace = (!xBar.isEmpty()) ? (2 * scrollbarPadding + scrollbarWidth + SCROLLBAR_LINE) : 0;
border = Rect(yBar.x - SCROLLBAR_LINE, scrollbarPadding, scrollbarWidth + 2 * SCROLLBAR_LINE, rect.height - 2 * scrollbarPadding - scrollSpace);
}
return border;
}
void ScrollableContainer::invalidateScrollbars()
{
Rect xBar = getXScrollbar();
Rect yBar = getYScrollbar();
Rect xBorder = getXBorder(xBar, yBar);
Rect yBorder = getYBorder(xBar, yBar);
// The two if statements ensure that the two sliders is invalidates thereby hides them, before they are set to size zero.
if (xSlider.getY() > xBorder.y)
{
xSlider.invalidate();
}
if (ySlider.getX() > yBorder.x)
{
ySlider.invalidate();
}
xSlider.setPosition(xBar.x, xBar.y, xBar.width, xBar.height);
ySlider.setPosition(yBar.x, yBar.y, yBar.width, yBar.height);
// x-/yBorder is given the coordinates zero and the witdh of the visiable area for the scrollable container,
// to ensure that the entire area where for the scrollable bars is and have been is invalidated correct.
xBorder.x = 0;
xBorder.width = rect.width;
yBorder.height = rect.height;
yBorder.y = 0;
if (!xBorder.isEmpty())
{
invalidateRect(xBorder);
}
if (!yBorder.isEmpty())
{
invalidateRect(yBorder);
}
}
bool ScrollableContainer::doScroll(int16_t deltaX, int16_t deltaY)
{
if (!deltaX && !deltaY)
{
return false;
}
bool couldScroll = false;
Rect contained = getContainedArea();
if (contained.width > rect.width)
{
if (deltaX > 0)
{
if (contained.x + deltaX > 0)
{
deltaX = -contained.x;
}
}
else if (deltaX < 0)
{
if (contained.right() + deltaX < rect.width)
{
deltaX = rect.width - contained.right();
}
}
}
else
{
deltaX = 0;
}
if (contained.height > rect.height)
{
if (deltaY > 0)
{
if (contained.y + deltaY > 0)
{
deltaY = -contained.y;
}
}
else if (deltaY < 0)
{
if (contained.bottom() + deltaY < rect.height)
{
deltaY = rect.height - contained.bottom();
}
}
}
else
{
deltaY = 0;
}
if (deltaX || deltaY)
{
scrolledXDistance += deltaX;
scrolledYDistance += deltaY;
moveChildrenRelative(deltaX, deltaY);
invalidateScrollbars();
couldScroll = true;
}
return couldScroll;
}
void ScrollableContainer::childGeometryChanged()
{
Rect contained = getContainedArea();
// If children are not aligned top left, make sure they are
if (contained.y > 0)
{
moveChildrenRelative(0, -contained.y);
}
if (contained.x > 0)
{
moveChildrenRelative(-contained.x, 0);
}
// Make sure we haven't scrolled below the bottom
if (contained.bottom() < rect.height)
{
int16_t deltaY = contained.bottom() - rect.height;
if (contained.y > deltaY)
{
deltaY = contained.y;
}
scrolledYDistance -= deltaY;
moveChildrenRelative(0, -deltaY);
}
// Make sure we haven't scrolled too far to the right
if (contained.right() < rect.width)
{
int deltaX = contained.right() - rect.width;
if (contained.x > deltaX)
{
deltaX = contained.x;
}
scrolledXDistance -= deltaX;
moveChildrenRelative(-deltaX, 0);
}
invalidateScrollbars();
}
void ScrollableContainer::add(Drawable& d)
{
remove(xSlider);
remove(ySlider);
Container::add(d);
Container::add(xSlider);
Container::add(ySlider);
}
Rect ScrollableContainer::getContainedArea() const
{
Drawable* d = firstChild;
Rect contained(0, 0, 0, 0);
Rect r(0, 0, rect.width, rect.height);
contained.expandToFit(r);
while (d)
{
if ((d != &xSlider) && (d != &ySlider) && (d->isVisible()))
{
contained.expandToFit(d->getRect());
}
d = d->getNextSibling();
}
return contained;
}
void ScrollableContainer::reset()
{
moveChildrenRelative(-scrolledXDistance, -scrolledYDistance);
scrolledXDistance = 0;
scrolledYDistance = 0;
invalidateScrollbars();
}
void ScrollableContainer::moveChildrenRelative(int16_t deltaX, int16_t deltaY)
{
Drawable* d = firstChild;
while (d)
{
if ((d != &xSlider) && (d != &ySlider))
{
d->moveRelative(deltaX, deltaY);
}
d = d->getNextSibling();
}
}
void ScrollableContainer::handleTickEvent()
{
if (animate)
{
// Calculate new position or stop animation
animationCounter++;
if (animationCounter <= scrollDuration)
{
// Calculate value in [beginningValue ; (beginningValue+targetValue)]
int16_t calculatedValue = EasingEquations::cubicEaseOut(animationCounter, beginningValue, targetValue, scrollDuration);
// Note: Result of "calculatedValue & 1" is compiler dependent for negative values of calculatedValue
if (calculatedValue % 2)
{
// Optimization: calculatedValue is odd, add 1/-1 to move drawables modulo 32 bits in framebuffer
calculatedValue += (calculatedValue > 0) ? 1 : -1;
}
// Convert to delta value relative to current X or Y
int16_t scrollX = (accelDirection == GestureEvent::SWIPE_VERTICAL) ? 0 : (calculatedValue - getContainedArea().x);
int16_t scrollY = (accelDirection == GestureEvent::SWIPE_HORIZONTAL) ? 0 : (calculatedValue - getContainedArea().y);
// Perform the actual animation step, stop animation if
// scrolling was not possible (doScroll invalidates children)
animate = doScroll(scrollX, scrollY);
}
else
{
animate = false;
}
if (!animate)
{
Application::getInstance()->unregisterTimerWidget(this);
animationCounter = 0;
}
}
}
void ScrollableContainer::setScrollbarsColor(colortype color)
{
scrollbarColor = color;
xSlider.setColor(scrollbarColor);
ySlider.setColor(scrollbarColor);
}
void ScrollableContainer::setScrollbarsAlpha(uint8_t alpha)
{
scrollbarAlpha = alpha;
xSlider.setAlpha(scrollbarAlpha);
ySlider.setAlpha(scrollbarAlpha);
}
void ScrollableContainer::setScrollbarPadding(uint8_t padding)
{
scrollbarPadding = padding;
}
void ScrollableContainer::setScrollbarWidth(uint8_t width)
{
scrollbarWidth = width;
}
void ScrollableContainer::setScrollbarsVisible(bool newVisible)
{
scrollbarsVisible = newVisible;
}
void ScrollableContainer::setScrollbarsPermanentlyVisible(bool permanentlyVisible /*= true*/)
{
scrollbarsPermanentlyVisible = permanentlyVisible;
xSlider.setVisible(true);
ySlider.setVisible(true);
invalidateScrollbars();
}
int16_t ScrollableContainer::getScrolledX() const
{
return scrolledXDistance;
}
int16_t ScrollableContainer::getScrolledY() const
{
return scrolledYDistance;
}
void ScrollableContainer::setScrollDurationSpeedup(uint16_t speedup)
{
scrollDurationSpeedup = MAX(1, speedup);
}
uint16_t ScrollableContainer::getScrollDurationSpeedup() const
{
return scrollDurationSpeedup;
}
void ScrollableContainer::setScrollDurationSlowdown(uint16_t slowdown)
{
scrollDurationSlowdown = MAX(1, slowdown);
}
uint16_t ScrollableContainer::getScrollDurationSlowdown() const
{
return scrollDurationSlowdown;
}
} // namespace touchgfx

View File

@ -0,0 +1,402 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <touchgfx/containers/SlideMenu.hpp>
namespace touchgfx
{
SlideMenu::SlideMenu()
: Container(),
onStateChangeButtonClicked(this, &SlideMenu::stateChangeButtonClickedHandler),
animationEndedCallback(this, &SlideMenu::animationEndedHandler),
stateChangedCallback(0),
stateChangedAnimationEndedCallback(0),
currentState(COLLAPSED),
expandDirection(EAST),
animationEquation(EasingEquations::cubicEaseInOut),
visiblePixelsWhenCollapsed(0),
hiddenPixelsWhenExpanded(0),
expandedStateTimeout(200),
expandedStateTimer(0),
animationDuration(10)
{
Application::getInstance()->registerTimerWidget(this);
stateChangeButton.setAction(onStateChangeButtonClicked);
menuContainer.setMoveAnimationEndedAction(animationEndedCallback);
Container::add(menuContainer);
menuContainer.add(background);
menuContainer.add(stateChangeButton);
}
SlideMenu::~SlideMenu()
{
Application::getInstance()->unregisterTimerWidget(this); //lint !e1551
}
void SlideMenu::setup(ExpandDirection newExpandDirection, const Bitmap& backgroundBMP, const Bitmap& stateChangeButtonBMP, const Bitmap& stateChangeButtonPressedBMP)
{
int16_t backgroundX = 0;
int16_t backgroundY = 0;
int16_t buttonX = 0;
int16_t buttonY = 0;
switch (newExpandDirection)
{
case SOUTH:
backgroundX = 0;
backgroundY = 0;
buttonX = (backgroundBMP.getWidth() - stateChangeButtonBMP.getWidth()) / 2;
buttonY = backgroundBMP.getHeight();
setVisiblePixelsWhenCollapsed(stateChangeButtonBMP.getHeight());
break;
case NORTH:
backgroundX = 0;
backgroundY = stateChangeButtonBMP.getHeight();
buttonX = (backgroundBMP.getWidth() - stateChangeButtonBMP.getWidth()) / 2;
buttonY = 0;
setVisiblePixelsWhenCollapsed(stateChangeButtonBMP.getHeight());
break;
case EAST:
backgroundX = 0;
backgroundY = 0;
buttonX = backgroundBMP.getWidth();
buttonY = (backgroundBMP.getHeight() - stateChangeButtonBMP.getHeight()) / 2;
setVisiblePixelsWhenCollapsed(stateChangeButtonBMP.getWidth());
break;
case WEST:
backgroundX = stateChangeButtonBMP.getWidth();
backgroundY = 0;
buttonX = 0;
buttonY = (backgroundBMP.getHeight() - stateChangeButtonBMP.getHeight()) / 2;
setVisiblePixelsWhenCollapsed(stateChangeButtonBMP.getWidth());
break;
default:
break;
}
setup(newExpandDirection, backgroundBMP, stateChangeButtonBMP, stateChangeButtonPressedBMP, backgroundX, backgroundY, buttonX, buttonY);
}
void SlideMenu::setup(ExpandDirection newExpandDirection, const Bitmap& backgroundBMP, int16_t backgroundX, int16_t backgroundY)
{
setExpandDirection(newExpandDirection);
background.setBitmap(backgroundBMP);
background.setXY(backgroundX, backgroundY);
Rect boundingRect = background.getRect();
// boundingRect.expandToFit(background.getRect());
menuContainer.setWidth(boundingRect.right());
menuContainer.setHeight(boundingRect.bottom());
setWidthHeight(menuContainer);
setExpandDirection(expandDirection);
setState(currentState);
invalidate();
}
void SlideMenu::setup(ExpandDirection newExpandDirection, const Bitmap& backgroundBMP, const Bitmap& stateChangeButtonBMP, const Bitmap& stateChangeButtonPressedBMP, int16_t backgroundX, int16_t backgroundY, int16_t stateChangeButtonX, int16_t stateChangeButtonY)
{
setExpandDirection(newExpandDirection);
background.setBitmap(backgroundBMP);
background.setXY(backgroundX, backgroundY);
stateChangeButton.setBitmaps(stateChangeButtonBMP, stateChangeButtonPressedBMP);
stateChangeButton.setXY(stateChangeButtonX, stateChangeButtonY);
Rect boundingRect(0, 0, 0, 0);
boundingRect.expandToFit(background.getRect());
boundingRect.expandToFit(stateChangeButton.getRect());
menuContainer.setWidth(boundingRect.right());
menuContainer.setHeight(boundingRect.bottom());
setWidthHeight(menuContainer);
setExpandDirection(expandDirection);
setState(currentState);
invalidate();
}
void SlideMenu::setExpandDirection(ExpandDirection newExpandDirection)
{
expandDirection = newExpandDirection;
setState(currentState);
}
SlideMenu::ExpandDirection SlideMenu::getExpandDirection() const
{
return expandDirection;
}
void SlideMenu::setVisiblePixelsWhenCollapsed(int16_t visiblePixels)
{
visiblePixelsWhenCollapsed = visiblePixels;
setState(currentState);
}
int16_t SlideMenu::getVisiblePixelsWhenCollapsed() const
{
return visiblePixelsWhenCollapsed;
}
void SlideMenu::setHiddenPixelsWhenExpanded(int16_t hiddenPixels)
{
hiddenPixelsWhenExpanded = hiddenPixels;
setState(currentState);
}
int16_t SlideMenu::getHiddenPixelsWhenExpanded() const
{
return hiddenPixelsWhenExpanded;
}
void SlideMenu::setExpandedStateTimeout(uint16_t timeout)
{
expandedStateTimeout = timeout;
}
uint16_t SlideMenu::getExpandedStateTimeout() const
{
return expandedStateTimeout;
}
void SlideMenu::setAnimationDuration(uint16_t duration)
{
animationDuration = duration;
}
uint16_t SlideMenu::getAnimationDuration() const
{
return animationDuration;
}
void SlideMenu::setAnimationEasingEquation(EasingEquation animationEasingEquation)
{
animationEquation = animationEasingEquation;
}
EasingEquation SlideMenu::getAnimationEasingEquation() const
{
return animationEquation;
}
void SlideMenu::setState(State newState)
{
if (newState == COLLAPSED)
{
menuContainer.moveTo(getCollapsedXCoordinate(), getCollapsedYCoordinate());
}
else
{
menuContainer.moveTo(getExpandedXCoordinate(), getExpandedYCoordinate());
}
currentState = newState;
}
void SlideMenu::animateToState(State newState)
{
if (animationDuration == 0)
{
setState(newState);
}
else if (newState != currentState)
{
if (currentState == COLLAPSED)
{
menuContainer.startMoveAnimation(getExpandedXCoordinate(), getExpandedYCoordinate(), animationDuration, animationEquation, animationEquation);
currentState = EXPANDED;
}
else
{
menuContainer.startMoveAnimation(getCollapsedXCoordinate(), getCollapsedYCoordinate(), animationDuration, animationEquation, animationEquation);
currentState = COLLAPSED;
}
// Disable stateChangeButton while animating
stateChangeButton.setTouchable(false);
}
}
SlideMenu::State SlideMenu::getState()
{
return currentState;
}
void SlideMenu::resetExpandedStateTimer()
{
expandedStateTimer = 0;
}
uint16_t SlideMenu::getExpandedStateTimer() const
{
return expandedStateTimer;
}
int16_t SlideMenu::getBackgroundX() const
{
return background.getX();
}
int16_t SlideMenu::getBackgroundY() const
{
return background.getY();
}
int16_t SlideMenu::getStateChangeButtonX() const
{
return stateChangeButton.getX();
}
int16_t SlideMenu::getStateChangeButtonY() const
{
return stateChangeButton.getY();
}
void SlideMenu::setStateChangedCallback(GenericCallback<const SlideMenu&>& callback)
{
stateChangedCallback = &callback;
}
void SlideMenu::setStateChangedAnimationEndedCallback(GenericCallback<const SlideMenu&>& callback)
{
stateChangedAnimationEndedCallback = &callback;
}
void SlideMenu::handleTickEvent()
{
if ((expandedStateTimeout != 0) && (currentState == EXPANDED) && !menuContainer.isMoveAnimationRunning())
{
expandedStateTimer++;
if (expandedStateTimer > expandedStateTimeout)
{
animateToState(COLLAPSED);
}
}
}
void SlideMenu::add(Drawable& d)
{
menuContainer.add(d);
}
void SlideMenu::remove(Drawable& d)
{
menuContainer.remove(d);
}
void SlideMenu::stateChangeButtonClickedHandler(const AbstractButton& /*button*/)
{
if (currentState == COLLAPSED)
{
animateToState(EXPANDED);
}
else
{
animateToState(COLLAPSED);
}
if ((stateChangedCallback != 0) && stateChangedCallback->isValid())
{
stateChangedCallback->execute(*this);
}
}
void SlideMenu::animationEndedHandler(const MoveAnimator<Container>& /*container*/)
{
resetExpandedStateTimer();
stateChangeButton.setTouchable(true);
if ((stateChangedAnimationEndedCallback != 0) && stateChangedAnimationEndedCallback->isValid())
{
stateChangedAnimationEndedCallback->execute(*this);
}
}
int16_t SlideMenu::getCollapsedXCoordinate()
{
switch (expandDirection)
{
case EAST:
return -menuContainer.getWidth() + visiblePixelsWhenCollapsed;
case WEST:
return getWidth() - visiblePixelsWhenCollapsed;
case SOUTH:
case NORTH:
default:
return 0;
}
}
int16_t SlideMenu::getCollapsedYCoordinate()
{
switch (expandDirection)
{
case SOUTH:
return -menuContainer.getHeight() + visiblePixelsWhenCollapsed;
case NORTH:
return getHeight() - visiblePixelsWhenCollapsed;
case EAST:
case WEST:
default:
return 0;
}
}
int16_t SlideMenu::getExpandedXCoordinate()
{
switch (expandDirection)
{
case EAST:
return -hiddenPixelsWhenExpanded;
case WEST:
return hiddenPixelsWhenExpanded;
case SOUTH:
case NORTH:
default:
return 0;
}
}
int16_t SlideMenu::getExpandedYCoordinate()
{
switch (expandDirection)
{
case SOUTH:
return -hiddenPixelsWhenExpanded;
case NORTH:
return hiddenPixelsWhenExpanded;
case EAST:
case WEST:
default:
return 0;
}
}
} // namespace touchgfx

View File

@ -0,0 +1,295 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <touchgfx/containers/Slider.hpp>
namespace touchgfx
{
Slider::Slider()
: Container(),
sliderOrientation(HORIZONTAL),
currentValue(0),
valueRangeMin(0),
valueRangeMax(1),
indicatorMinPosition(0),
indicatorMaxPosition(1),
startValueCallback(0),
stopValueCallback(0),
newValueCallback(0)
{
setTouchable(true);
// The backgroundSelectedViewPort is a container into which the bitmap for the "filled" background
// is placed. Containers are viewports, so the dimensions of this container controls how
// much of the filled background is visible.
backgroundSelectedViewPort.add(backgroundSelected);
Container::add(background);
Container::add(backgroundSelectedViewPort);
Container::add(indicator);
// Default value range
Slider::setValueRange(0, 100);
}
void Slider::setBitmaps(const Bitmap& sliderBackground, const Bitmap& sliderBackgroundSelected, const Bitmap& indicatorBitmap)
{
assert(sliderBackground.getWidth() == sliderBackgroundSelected.getWidth() &&
sliderBackground.getHeight() == sliderBackgroundSelected.getHeight() &&
"Slider::setBitmaps - background and backgroundFilled must have same dimensions");
background.setBitmap(sliderBackground);
backgroundSelected.setBitmap(sliderBackgroundSelected);
indicator.setBitmap(indicatorBitmap);
backgroundSelectedViewPort.setWidthHeight(backgroundSelected);
}
void Slider::setBitmaps(const BitmapId sliderBackground, const BitmapId sliderBackgroundSelected, const BitmapId indicatorBitmap)
{
setBitmaps(Bitmap(sliderBackground), Bitmap(sliderBackgroundSelected), Bitmap(indicatorBitmap));
}
void Slider::setupHorizontalSlider(uint16_t backgroundX, uint16_t backgroundY, uint16_t indicatorY, uint16_t indicatorMinX, uint16_t indicatorMaxX)
{
assert(indicatorMinX < indicatorMaxX && "Slider::setupHorizontalSlider - indicatorMinX must be smaller than indicatorMaxX");
sliderOrientation = HORIZONTAL;
background.setXY(backgroundX, backgroundY);
backgroundSelectedViewPort.setXY(backgroundX, backgroundY);
backgroundSelected.setXY(0, 0);
indicator.setY(indicatorY);
uint16_t backgroundWidth = backgroundX + static_cast<uint16_t>(background.getWidth());
uint16_t indicatorWidth = indicatorMaxX + static_cast<uint16_t>(indicator.getWidth());
int16_t newWidth = static_cast<int16_t>(MAX(backgroundWidth, indicatorWidth));
uint16_t backgroundHeight = backgroundY + static_cast<uint16_t>(background.getHeight());
uint16_t indicatorHeight = indicatorY + static_cast<uint16_t>(indicator.getHeight());
int16_t newHeight = static_cast<int16_t>(MAX(backgroundHeight, indicatorHeight));
indicatorMinPosition = indicatorMinX;
indicatorMaxPosition = indicatorMaxX;
setWidthHeight(newWidth, newHeight);
setValue(currentValue);
}
void Slider::setupVerticalSlider(uint16_t backgroundX, uint16_t backgroundY, uint16_t indicatorX, uint16_t indicatorMinY, uint16_t indicatorMaxY)
{
assert(indicatorMinY < indicatorMaxY && "Slider::setupVerticalSlider - indicatorMinY must be smaller than indicatorMaxY");
sliderOrientation = VERTICAL;
background.setXY(backgroundX, backgroundY);
backgroundSelectedViewPort.setXY(backgroundX, backgroundY);
indicator.setX(indicatorX);
uint16_t backgroundWidth = backgroundX + static_cast<uint16_t>(background.getWidth());
uint16_t indicatorWidth = indicatorX + static_cast<uint16_t>(indicator.getWidth());
int16_t newWidth = static_cast<int16_t>(MAX(backgroundWidth, indicatorWidth));
uint16_t backgroundHeight = backgroundY + static_cast<uint16_t>(background.getHeight());
uint16_t indicatorHeight = indicatorMaxY + static_cast<uint16_t>(indicator.getHeight());
int16_t newHeight = static_cast<int16_t>(MAX(backgroundHeight, indicatorHeight));
indicatorMinPosition = indicatorMinY;
indicatorMaxPosition = indicatorMaxY;
setWidthHeight(newWidth, newHeight);
setValue(currentValue);
}
void Slider::setValue(int value)
{
updateIndicatorPosition(valueToPosition(value));
}
void Slider::handleClickEvent(const ClickEvent& evt)
{
if ((evt.getType() == ClickEvent::PRESSED) || (evt.getType() == ClickEvent::RELEASED))
{
// Communicate the start value if a listener is registered
if ((evt.getType() == ClickEvent::PRESSED) && (startValueCallback != 0) && startValueCallback->isValid())
{
startValueCallback->execute(*this, currentValue);
}
if (sliderOrientation == HORIZONTAL)
{
updateIndicatorPosition(evt.getX() - getIndicatorRadius());
}
else
{
updateIndicatorPosition(evt.getY() - getIndicatorRadius());
}
// Communicate the stop value if a listener is registered
if ((evt.getType() == ClickEvent::RELEASED) && (stopValueCallback != 0) && stopValueCallback->isValid())
{
stopValueCallback->execute(*this, currentValue);
}
}
}
void Slider::handleDragEvent(const DragEvent& evt)
{
if (sliderOrientation == HORIZONTAL)
{
updateIndicatorPosition(evt.getNewX() - getIndicatorRadius());
}
else
{
updateIndicatorPosition(evt.getNewY() - getIndicatorRadius());
}
}
int16_t Slider::valueToPosition(int value) const
{
value = MIN(valueRangeMax, value);
value = MAX(value, valueRangeMin);
int coordinateOffset = ((value - valueRangeMin) * (getIndicatorPositionRangeSize() + 1)) / getValueRangeSize();
int result = indicatorMinPosition + coordinateOffset;
if (sliderOrientation == VERTICAL)
{
// Vertical slider grows as the position decreases so invert the coordinate
result = indicatorMinPosition + (indicatorMaxPosition - result);
}
return result;
}
int Slider::positionToValue(int16_t position) const
{
int result;
if (position == indicatorMinPosition)
{
// Ensure that min coordinate always results in min value
result = valueRangeMin;
}
else if (position == indicatorMaxPosition)
{
// Ensure that max coordinate always results in max value
result = valueRangeMax;
}
else
{
int rounding = getIndicatorPositionRangeSize() / 2;
int valueOffset = (((position - indicatorMinPosition) * getValueRangeSize()) + rounding) / getIndicatorPositionRangeSize();
result = valueRangeMin + valueOffset;
}
if (sliderOrientation == VERTICAL)
{
// Vertical slider grows as the position decreases so invert the value
result = valueRangeMin + (valueRangeMax - result);
}
return result;
}
void Slider::updateIndicatorPosition(int16_t position)
{
// Cut off positions outside the slider area
position = MAX(position, indicatorMinPosition);
position = MIN(position, indicatorMaxPosition);
if (sliderOrientation == HORIZONTAL)
{
indicator.moveTo(position, indicator.getY());
backgroundSelectedViewPort.invalidate();
backgroundSelectedViewPort.setWidth((position - backgroundSelectedViewPort.getX()) + getIndicatorRadius());
backgroundSelectedViewPort.invalidate();
}
else
{
indicator.moveTo(indicator.getX(), position);
backgroundSelectedViewPort.invalidate();
int16_t newViewPortHeight = background.getRect().bottom() - (position + getIndicatorRadius());
backgroundSelectedViewPort.setPosition(backgroundSelectedViewPort.getX(), position + getIndicatorRadius(), backgroundSelectedViewPort.getWidth(), newViewPortHeight);
backgroundSelected.setY(-(backgroundSelected.getHeight() - newViewPortHeight));
backgroundSelectedViewPort.invalidate();
}
currentValue = positionToValue(position);
// Communicate the new value if a listener is registered
if ((newValueCallback != 0) && newValueCallback->isValid())
{
newValueCallback->execute(*this, currentValue);
}
}
uint16_t Slider::getIndicatorRadius() const
{
uint16_t result;
if (sliderOrientation == HORIZONTAL)
{
result = indicator.getWidth() / 2;
}
else
{
result = indicator.getHeight() / 2;
}
return result;
}
void Slider::setValueRange(int minValue, int maxValue, int newValue)
{
assert(minValue < maxValue && "Slider::setValueRange - minValue must be smaller than maxValue");
valueRangeMin = minValue;
valueRangeMax = maxValue;
setValue(newValue);
}
void Slider::setValueRange(int minValue, int maxValue)
{
int newValue = currentValue;
if (currentValue < minValue)
{
newValue = minValue;
}
else if (currentValue > maxValue)
{
newValue = maxValue;
}
setValueRange(minValue, maxValue, newValue);
}
int Slider::getIndicatorPositionRangeSize() const
{
return indicatorMaxPosition - indicatorMinPosition;
}
int Slider::getValueRangeSize() const
{
return valueRangeMax - valueRangeMin;
}
} // namespace touchgfx

View File

@ -0,0 +1,400 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <touchgfx/EasingEquations.hpp>
#include <touchgfx/containers/SwipeContainer.hpp>
namespace touchgfx
{
SwipeContainer::SwipeContainer()
: Container(),
currentState(NO_ANIMATION),
numberOfPages(0),
animationCounter(0),
swipeCutoff(80),
dragX(0),
animateDistance(0),
startX(0),
currentPage(0),
endElasticWidth(30),
pages(EAST),
pageIndicator()
{
Application::getInstance()->registerTimerWidget(this);
setTouchable(true);
Container::add(pages);
Container::add(pageIndicator);
}
SwipeContainer::~SwipeContainer()
{
Application::getInstance()->unregisterTimerWidget(this);
}
void SwipeContainer::add(Drawable& page)
{
pages.add(page);
numberOfPages++;
pageIndicator.setNumberOfPages(numberOfPages);
setWidthHeight(page);
}
void SwipeContainer::remove(Drawable& page)
{
Drawable* tmp = pages.getFirstChild();
if (!numberOfPages)
{
return;
}
// looks for the child matching page
// to ensure that the page indicator only counts down if a page is removed
while (tmp)
{
if (tmp == &page)
{
pages.remove(page);
numberOfPages--;
if (!numberOfPages)
{
setWidthHeight(0, 0);
}
else
{
pageIndicator.setNumberOfPages(numberOfPages);
}
return;
}
else
{
tmp = tmp->getNextSibling();
}
}
}
void SwipeContainer::setEndSwipeElasticWidth(uint16_t width)
{
endElasticWidth = width;
}
void SwipeContainer::setSwipeCutoff(uint16_t cutoff)
{
swipeCutoff = cutoff;
}
void SwipeContainer::setPageIndicatorBitmaps(const Bitmap& normalPage, const Bitmap& highlightedPage)
{
pageIndicator.setBitmaps(normalPage, highlightedPage);
}
void SwipeContainer::setPageIndicatorXY(int16_t x, int16_t y)
{
pageIndicator.setXY(x, y);
}
void SwipeContainer::setPageIndicatorXYWithCenteredX(int16_t x, int16_t y)
{
pageIndicator.setXY(x - pageIndicator.getWidth() / 2, y);
}
void SwipeContainer::setSelectedPage(uint8_t pageIndex)
{
currentPage = pageIndex;
pageIndicator.setHighlightPosition(currentPage);
adjustPages();
}
uint8_t SwipeContainer::getSelectedPage() const
{
return currentPage;
}
void SwipeContainer::handleTickEvent()
{
if (currentState == ANIMATE_SWIPE_CANCELLED_LEFT)
{
animateSwipeCancelledLeft();
}
else if (currentState == ANIMATE_SWIPE_CANCELLED_RIGHT)
{
animateSwipeCancelledRight();
}
else if (currentState == ANIMATE_LEFT)
{
animateLeft();
}
else if (currentState == ANIMATE_RIGHT)
{
animateRight();
}
}
void SwipeContainer::handleClickEvent(const ClickEvent& evt)
{
// If an animation is already in progress do not
// react to clicks
if (currentState != NO_ANIMATION)
{
return;
}
if (evt.getType() == ClickEvent::RELEASED)
{
// Save current position for use during animation
animateDistance = dragX;
startX = pages.getX();
if (dragX < 0)
{
if (currentPage == getNumberOfPages() - 1 || dragX > -swipeCutoff)
{
currentState = ANIMATE_SWIPE_CANCELLED_LEFT;
}
else
{
currentState = ANIMATE_LEFT;
}
}
else if (dragX > 0)
{
if (currentPage == 0 || dragX < swipeCutoff)
{
currentState = ANIMATE_SWIPE_CANCELLED_RIGHT;
}
else
{
currentState = ANIMATE_RIGHT;
}
}
}
}
void SwipeContainer::handleDragEvent(const DragEvent& evt)
{
// If an animation is already in progress do not
// react to drags
if (currentState != NO_ANIMATION)
{
return;
}
dragX += evt.getDeltaX();
// Do not show too much background next to end pages
if (currentPage == 0 && dragX > endElasticWidth)
{
dragX = static_cast<int16_t>(endElasticWidth);
}
else if (currentPage == getNumberOfPages() - 1 && dragX < -endElasticWidth)
{
dragX = -static_cast<int16_t>(endElasticWidth);
}
adjustPages();
}
void SwipeContainer::handleGestureEvent(const GestureEvent& evt)
{
// Do not accept gestures while animating
if (currentState != NO_ANIMATION)
{
return;
}
if (evt.getType() == GestureEvent::SWIPE_HORIZONTAL)
{
// Save current position for use during animation
animateDistance = dragX;
startX = pages.getX();
if (evt.getVelocity() < 0 && currentPage < getNumberOfPages() - 1)
{
currentState = ANIMATE_LEFT;
}
else if (evt.getVelocity() > 0 && currentPage > 0)
{
currentState = ANIMATE_RIGHT;
}
}
}
void SwipeContainer::adjustPages()
{
pages.moveTo(-static_cast<int16_t>(currentPage * getWidth()) + dragX, 0);
}
void SwipeContainer::animateSwipeCancelledLeft()
{
uint8_t duration = 14;
if (animationCounter <= duration)
{
int16_t delta = EasingEquations::backEaseOut(animationCounter, 0, -animateDistance, duration);
dragX = animateDistance + delta;
adjustPages();
}
else
{
// Final step: stop the animation
currentState = NO_ANIMATION;
animationCounter = 0;
dragX = 0;
adjustPages();
}
animationCounter++;
}
void SwipeContainer::animateSwipeCancelledRight()
{
uint8_t duration = 14;
if (animationCounter <= duration)
{
int16_t delta = EasingEquations::backEaseOut(animationCounter, 0, animateDistance, duration);
dragX = animateDistance - delta;
adjustPages();
}
else
{
// Final step: stop the animation
currentState = NO_ANIMATION;
animationCounter = 0;
dragX = 0;
adjustPages();
}
animationCounter++;
}
void SwipeContainer::animateLeft()
{
uint8_t duration = 10;
if (animationCounter <= duration)
{
int16_t delta = EasingEquations::cubicEaseOut(animationCounter, 0, getWidth() + animateDistance, duration);
dragX = animateDistance - delta;
adjustPages();
}
else
{
// Final step: stop the animation
currentState = NO_ANIMATION;
animationCounter = 0;
currentPage++;
dragX = 0;
adjustPages();
pageIndicator.goRight();
}
animationCounter++;
}
void SwipeContainer::animateRight()
{
uint8_t duration = 10;
if (animationCounter <= duration)
{
int16_t delta = EasingEquations::cubicEaseOut(animationCounter, 0, getWidth() - animateDistance, duration);
dragX = animateDistance + delta;
adjustPages();
}
else
{
// Final step: stop the animation
currentState = NO_ANIMATION;
animationCounter = 0;
currentPage--;
dragX = 0;
adjustPages();
pageIndicator.goLeft();
}
animationCounter++;
}
SwipeContainer::PageIndicator::PageIndicator()
: Container(),
numberOfPages(0),
currentPage(0)
{
unselectedPages.setXY(0, 0);
selectedPage.setXY(0, 0);
Container::add(unselectedPages);
Container::add(selectedPage);
}
void SwipeContainer::PageIndicator::setNumberOfPages(uint8_t size)
{
numberOfPages = size;
assert(numberOfPages > 0 && "At least one dot is needed");
numberOfPages = size;
if (unselectedPages.getBitmapId() != BITMAP_INVALID)
{
int dotWidth = Bitmap(unselectedPages.getBitmap()).getWidth();
unselectedPages.setWidth(dotWidth * size);
// adjust size of container according to the actual bitmaps
setWidthHeight(unselectedPages);
setHighlightPosition(currentPage = 0);
invalidate();
}
}
void SwipeContainer::PageIndicator::setBitmaps(const Bitmap& normalPage, const Bitmap& highlightedPage)
{
selectedPage.setBitmap(highlightedPage);
unselectedPages.setBitmap(normalPage);
if (numberOfPages > 0)
{
setNumberOfPages(numberOfPages);
}
}
void SwipeContainer::PageIndicator::goRight()
{
setHighlightPosition(currentPage = (currentPage + 1) % numberOfPages);
}
void SwipeContainer::PageIndicator::goLeft()
{
setHighlightPosition(currentPage = (currentPage + numberOfPages - 1) % numberOfPages);
}
void SwipeContainer::PageIndicator::setHighlightPosition(uint8_t index)
{
currentPage = index;
// note that index is unsigned
if (index < numberOfPages)
{
int dotWidth = Bitmap(unselectedPages.getBitmap()).getWidth();
selectedPage.setX(index * dotWidth);
}
invalidate();
}
} // namespace touchgfx

View File

@ -0,0 +1,288 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <touchgfx/containers/ZoomAnimationImage.hpp>
namespace touchgfx
{
ZoomAnimationImage::ZoomAnimationImage()
: Container(),
currentState(NO_ANIMATION),
animationCounter(0),
zoomAnimationDelay(0),
currentZoomMode(FIXED_LEFT_AND_TOP),
zoomAnimationStartWidth(0),
zoomAnimationStartHeight(0),
zoomAnimationEndWidth(0),
zoomAnimationEndHeight(0),
zoomAnimationStartX(0),
zoomAnimationStartY(0),
zoomAnimationDeltaX(0),
zoomAnimationDeltaY(0),
moveAnimationEndX(0),
moveAnimationEndY(0),
animationDuration(0),
zoomAnimationWidthEquation(EasingEquations::linearEaseNone),
zoomAnimationHeightEquation(EasingEquations::linearEaseNone),
moveAnimationXEquation(EasingEquations::linearEaseNone),
moveAnimationYEquation(EasingEquations::linearEaseNone),
animationEndedAction(0)
{
image.setXY(0, 0);
image.setVisible(false);
scalableImage.setScalingAlgorithm(ScalableImage::NEAREST_NEIGHBOR);
scalableImage.setXY(0, 0);
scalableImage.setVisible(false);
Container::add(image);
Container::add(scalableImage);
}
void ZoomAnimationImage::startZoomAnimation(int16_t endWidth, int16_t endHeight, uint16_t duration, ZoomMode zoomMode, EasingEquation widthProgressionEquation, EasingEquation heightProgressionEquation)
{
setCurrentState(ANIMATE_ZOOM);
startTimerAndSetParameters(endWidth, endHeight, duration, zoomMode, widthProgressionEquation, heightProgressionEquation);
}
void ZoomAnimationImage::startZoomAndMoveAnimation(int16_t endX, int16_t endY, int16_t endWidth, int16_t endHeight, uint16_t duration, ZoomMode zoomMode, EasingEquation xProgressionEquation, EasingEquation yProgressionEquation, EasingEquation widthProgressionEquation, EasingEquation heightProgressionEquation)
{
moveAnimationEndX = endX;
moveAnimationEndY = endY;
moveAnimationXEquation = xProgressionEquation;
moveAnimationYEquation = yProgressionEquation;
setCurrentState(ANIMATE_ZOOM_AND_MOVE);
startTimerAndSetParameters(endWidth, endHeight, duration, zoomMode, widthProgressionEquation, heightProgressionEquation);
}
void ZoomAnimationImage::cancelZoomAnimation()
{
Application::getInstance()->unregisterTimerWidget(this);
setCurrentState(NO_ANIMATION);
}
void ZoomAnimationImage::handleTickEvent()
{
if ((currentState == ANIMATE_ZOOM) || (currentState == ANIMATE_ZOOM_AND_MOVE))
{
animationCounter++;
if (animationCounter >= zoomAnimationDelay)
{
// Adjust the used animationCounter for the startup delay
uint32_t actualAnimationCounter = animationCounter - zoomAnimationDelay;
int16_t deltaWidth = zoomAnimationWidthEquation(actualAnimationCounter, 0, zoomAnimationEndWidth - zoomAnimationStartWidth, animationDuration);
int16_t deltaHeight = zoomAnimationHeightEquation(actualAnimationCounter, 0, zoomAnimationEndHeight - zoomAnimationStartHeight, animationDuration);
setWidthHeight(zoomAnimationStartWidth + deltaWidth, zoomAnimationStartHeight + deltaHeight);
int16_t deltaX;
int16_t deltaY;
if (currentState == ANIMATE_ZOOM_AND_MOVE)
{
deltaX = moveAnimationXEquation(actualAnimationCounter, 0, (moveAnimationEndX - zoomAnimationStartX) + zoomAnimationDeltaX, animationDuration);
deltaY = moveAnimationYEquation(actualAnimationCounter, 0, (moveAnimationEndY - zoomAnimationStartY) + zoomAnimationDeltaY, animationDuration);
}
else
{
deltaX = zoomAnimationWidthEquation(actualAnimationCounter, 0, zoomAnimationDeltaX, animationDuration);
deltaY = zoomAnimationHeightEquation(actualAnimationCounter, 0, zoomAnimationDeltaY, animationDuration);
}
moveTo(zoomAnimationStartX + deltaX, zoomAnimationStartY + deltaY);
if (animationCounter >= (uint32_t)(zoomAnimationDelay + animationDuration))
{
cancelZoomAnimation();
if (animationEndedAction && animationEndedAction->isValid())
{
animationEndedAction->execute(*this);
}
}
}
}
}
void ZoomAnimationImage::setBitmaps(const Bitmap& smallBitmap, const Bitmap& largeBitmap)
{
smallBmp = smallBitmap;
largeBmp = largeBitmap;
scalableImage.setBitmap(largeBmp);
ZoomAnimationImage::setWidthHeight(largeBmp);
}
void ZoomAnimationImage::setWidth(int16_t width)
{
invalidate();
Container::setWidth(width);
updateRenderingMethod();
}
void ZoomAnimationImage::setHeight(int16_t height)
{
invalidate();
Container::setHeight(height);
updateRenderingMethod();
}
void ZoomAnimationImage::setDimension(int16_t width, int16_t height)
{
Container::setWidthHeight(width, height);
}
void ZoomAnimationImage::setScalingMode(ScalableImage::ScalingAlgorithm mode)
{
scalableImage.setScalingAlgorithm(mode);
}
ScalableImage::ScalingAlgorithm ZoomAnimationImage::getScalingMode()
{
return scalableImage.getScalingAlgorithm();
}
void ZoomAnimationImage::setAlpha(const uint8_t newAlpha)
{
image.setAlpha(newAlpha);
scalableImage.setAlpha(newAlpha);
}
uint8_t ZoomAnimationImage::getAlpha() const
{
return image.getAlpha();
}
void ZoomAnimationImage::setAnimationDelay(uint16_t delay)
{
zoomAnimationDelay = delay;
}
uint16_t ZoomAnimationImage::getAnimationDelay() const
{
return zoomAnimationDelay;
}
bool ZoomAnimationImage::isZoomAnimationRunning() const
{
return currentState != NO_ANIMATION;
}
void ZoomAnimationImage::updateRenderingMethod()
{
if ((smallBmp.getWidth() == getWidth()) && (smallBmp.getHeight() == getHeight()))
{
image.setVisible(true);
scalableImage.setVisible(false);
image.setBitmap(smallBmp);
image.invalidate();
scalableImage.invalidate();
}
else if ((largeBmp.getWidth() == getWidth()) && (largeBmp.getHeight() == getHeight()))
{
image.setVisible(true);
scalableImage.setVisible(false);
image.setBitmap(largeBmp);
image.invalidate();
scalableImage.invalidate();
}
else
{
image.setVisible(false);
image.invalidate();
scalableImage.setVisible(true);
scalableImage.setWidthHeight(*this);
scalableImage.invalidate();
}
}
void ZoomAnimationImage::setCurrentState(States state)
{
currentState = state;
animationCounter = 0;
}
void ZoomAnimationImage::startTimerAndSetParameters(int16_t endWidth, int16_t endHeight, uint16_t duration, ZoomMode zoomMode, EasingEquation widthProgressionEquation, EasingEquation heightProgressionEquation)
{
Application::getInstance()->registerTimerWidget(this);
currentZoomMode = zoomMode;
zoomAnimationStartX = getX();
zoomAnimationStartY = getY();
zoomAnimationStartWidth = getWidth();
zoomAnimationStartHeight = getHeight();
zoomAnimationEndWidth = endWidth;
zoomAnimationEndHeight = endHeight;
animationDuration = duration;
zoomAnimationWidthEquation = widthProgressionEquation;
zoomAnimationHeightEquation = heightProgressionEquation;
updateZoomAnimationDeltaXY();
if (zoomAnimationDelay == 0 && animationDuration == 0)
{
handleTickEvent(); // Finish the zoom and move operation immediately
}
}
void ZoomAnimationImage::updateZoomAnimationDeltaXY()
{
zoomAnimationDeltaX = zoomAnimationStartWidth - zoomAnimationEndWidth;
zoomAnimationDeltaY = zoomAnimationStartHeight - zoomAnimationEndHeight;
switch (currentZoomMode)
{
case FIXED_CENTER:
zoomAnimationDeltaX /= 2;
zoomAnimationDeltaY /= 2;
break;
case FIXED_LEFT:
zoomAnimationDeltaX = 0;
zoomAnimationDeltaY /= 2;
break;
case FIXED_RIGHT:
zoomAnimationDeltaY /= 2;
break;
case FIXED_TOP:
zoomAnimationDeltaX /= 2;
zoomAnimationDeltaY = 0;
break;
case FIXED_BOTTOM:
zoomAnimationDeltaX /= 2;
break;
case FIXED_LEFT_AND_TOP:
zoomAnimationDeltaX = 0;
zoomAnimationDeltaY = 0;
break;
case FIXED_RIGHT_AND_TOP:
zoomAnimationDeltaY = 0;
break;
case FIXED_LEFT_AND_BOTTOM:
zoomAnimationDeltaX = 0;
break;
case FIXED_RIGHT_AND_BOTTOM:
break;
default:
break;
}
}
} // namespace touchgfx

View File

@ -0,0 +1,71 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <touchgfx/containers/clock/AbstractClock.hpp>
namespace touchgfx
{
AbstractClock::AbstractClock()
: Container(),
currentHour(0),
currentMinute(0),
currentSecond(0)
{
}
void AbstractClock::setTime24Hour(uint8_t hour, uint8_t minute, uint8_t second)
{
currentHour = hour % 24;
currentMinute = minute % 60;
currentSecond = second % 60;
updateClock();
}
void AbstractClock::setTime12Hour(uint8_t hour, uint8_t minute, uint8_t second, bool am)
{
setTime24Hour((hour % 12) + (am ? 0 : 12), minute, second);
}
uint8_t AbstractClock::getCurrentHour() const
{
return currentHour;
}
uint8_t AbstractClock::getCurrentHour24() const
{
return currentHour;
}
uint8_t AbstractClock::getCurrentHour12() const
{
return ((currentHour + 11) % 12) + 1;
}
bool AbstractClock::getCurrentAM() const
{
return currentHour < 12;
}
uint8_t AbstractClock::getCurrentMinute() const
{
return currentMinute;
}
uint8_t AbstractClock::getCurrentSecond() const
{
return currentSecond;
}
} // namespace touchgfx

View File

@ -0,0 +1,243 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <touchgfx/containers/clock/AnalogClock.hpp>
namespace touchgfx
{
AnalogClock::AnalogClock()
: AbstractClock(),
animationEquation(EasingEquations::linearEaseNone),
animationDuration(0),
clockRotationCenterX(0),
clockRotationCenterY(0),
lastHour(0),
lastMinute(0),
lastSecond(0),
hourHandMinuteCorrectionActive(false),
minuteHandSecondCorrectionActive(false)
{
AnalogClock::add(background);
hourHand.updateZAngle(0.f);
minuteHand.updateZAngle(0.f);
secondHand.updateZAngle(0.f);
hourHand.setVisible(false);
minuteHand.setVisible(false);
secondHand.setVisible(false);
}
void AnalogClock::setBackground(const BitmapId backgroundBitmapId)
{
setBackground(backgroundBitmapId, Bitmap(backgroundBitmapId).getWidth() / 2, Bitmap(backgroundBitmapId).getHeight() / 2);
}
void AnalogClock::setBackground(const BitmapId backgroundBitmapId, const int16_t rotationCenterX, const int16_t rotationCenterY)
{
background.setBitmap(Bitmap(backgroundBitmapId));
setWidthHeight(background);
setRotationCenter(rotationCenterX, rotationCenterY);
}
void AnalogClock::setRotationCenter(int16_t rotationCenterX, int16_t rotationCenterY)
{
clockRotationCenterX = rotationCenterX;
clockRotationCenterY = rotationCenterY;
}
void AnalogClock::setupHourHand(const BitmapId hourHandBitmapId, int16_t rotationCenterX, int16_t rotationCenterY)
{
setupHand(hourHand, hourHandBitmapId, rotationCenterX, rotationCenterY);
}
void AnalogClock::setupMinuteHand(const BitmapId minuteHandBitmapId, int16_t rotationCenterX, int16_t rotationCenterY)
{
setupHand(minuteHand, minuteHandBitmapId, rotationCenterX, rotationCenterY);
}
void AnalogClock::setupSecondHand(const BitmapId secondHandBitmapId, int16_t rotationCenterX, int16_t rotationCenterY)
{
setupHand(secondHand, secondHandBitmapId, rotationCenterX, rotationCenterY);
}
void AnalogClock::setupHand(TextureMapper& hand, const BitmapId bitmapId, int16_t rotationCenterX, int16_t rotationCenterY)
{
remove(hand);
hand.setBitmap(Bitmap(bitmapId));
hand.setWidthHeight(*this);
hand.setXY(0, 0);
hand.setBitmapPosition(clockRotationCenterX - rotationCenterX, clockRotationCenterY - rotationCenterY);
hand.setCameraDistance(300.0f);
hand.setOrigo((float)clockRotationCenterX, (float)clockRotationCenterY, hand.getCameraDistance());
hand.setCamera(hand.getOrigoX(), hand.getOrigoY());
hand.setRenderingAlgorithm(TextureMapper::BILINEAR_INTERPOLATION);
add(hand);
hand.setVisible(true);
}
void AnalogClock::initializeTime24Hour(uint8_t hour, uint8_t minute, uint8_t second)
{
lastHour = 255;
lastMinute = 255;
lastSecond = 255;
// Disable animation and set time
uint16_t tempAnimationDuration = animationDuration;
animationDuration = 1;
setTime24Hour(hour, minute, second);
animationDuration = tempAnimationDuration;
}
void AnalogClock::initializeTime12Hour(uint8_t hour, uint8_t minute, uint8_t second, bool am)
{
initializeTime24Hour((hour % 12) + (am ? 0 : 12), minute, second);
}
void AnalogClock::setAlpha(uint8_t newAlpha)
{
background.setAlpha(newAlpha);
hourHand.setAlpha(newAlpha);
minuteHand.setAlpha(newAlpha);
secondHand.setAlpha(newAlpha);
}
uint8_t AnalogClock::getAlpha() const
{
return background.getAlpha();
}
void AnalogClock::updateClock()
{
// Make sure that animating to 0 will move from left to right
if (lastHour != 0 && currentHour == 0)
{
hourHand.updateZAngle(hourHand.getZAngle() - (2 * PI));
}
if (lastMinute != 0 && currentMinute == 0)
{
minuteHand.updateZAngle(minuteHand.getZAngle() - (2 * PI));
}
if (lastSecond != 0 && currentSecond == 0)
{
secondHand.updateZAngle(secondHand.getZAngle() - (2 * PI));
}
float newHandAngle;
// Move hour hand
if (hourHand.isVisible() && ((currentHour != lastHour) || (hourHandMinuteCorrectionActive && (currentMinute != lastMinute))))
{
newHandAngle = convertHandValueToAngle(12, currentHour, hourHandMinuteCorrectionActive ? currentMinute : 0);
if (animationEnabled() && !hourHand.isTextureMapperAnimationRunning())
{
hourHand.setupAnimation(AnimationTextureMapper::Z_ROTATION, newHandAngle, animationDuration, 0, animationEquation);
hourHand.startAnimation();
}
else
{
if (animationEnabled())
{
hourHand.cancelAnimationTextureMapperAnimation();
}
hourHand.updateZAngle(newHandAngle);
}
}
// Move minute hand
if (minuteHand.isVisible() && ((currentMinute != lastMinute) || (minuteHandSecondCorrectionActive && (currentSecond != lastSecond))))
{
newHandAngle = convertHandValueToAngle(60, currentMinute, minuteHandSecondCorrectionActive ? currentSecond : 0);
if (animationEnabled() && !minuteHand.isTextureMapperAnimationRunning())
{
minuteHand.setupAnimation(AnimationTextureMapper::Z_ROTATION, newHandAngle, animationDuration, 0, animationEquation);
minuteHand.startAnimation();
}
else
{
if (animationEnabled())
{
minuteHand.cancelAnimationTextureMapperAnimation();
}
minuteHand.updateZAngle(newHandAngle);
}
}
// Move second hand
if (secondHand.isVisible() && (currentSecond != lastSecond))
{
newHandAngle = convertHandValueToAngle(60, currentSecond);
if (animationEnabled() && !secondHand.isTextureMapperAnimationRunning())
{
secondHand.setupAnimation(AnimationTextureMapper::Z_ROTATION, newHandAngle, animationDuration, 0, animationEquation);
secondHand.startAnimation();
}
else
{
if (animationEnabled())
{
secondHand.cancelAnimationTextureMapperAnimation();
}
secondHand.updateZAngle(newHandAngle);
}
}
lastHour = currentHour;
lastMinute = currentMinute;
lastSecond = currentSecond;
}
float AnalogClock::convertHandValueToAngle(uint8_t steps, uint8_t handValue, uint8_t secondHandValue /*= 0*/) const
{
return ((handValue / (float)steps) + (secondHandValue / (steps * 60.f))) * 2 * PI;
}
void AnalogClock::setHourHandMinuteCorrection(bool active)
{
hourHandMinuteCorrectionActive = active;
setTime24Hour(getCurrentHour(), getCurrentMinute(), getCurrentSecond());
}
bool AnalogClock::getHourHandMinuteCorrection() const
{
return hourHandMinuteCorrectionActive;
}
void AnalogClock::setMinuteHandSecondCorrection(bool active)
{
minuteHandSecondCorrectionActive = active;
setTime24Hour(getCurrentHour(), getCurrentMinute(), getCurrentSecond());
}
bool AnalogClock::getMinuteHandSecondCorrection() const
{
return minuteHandSecondCorrectionActive;
}
bool AnalogClock::animationEnabled() const
{
return animationDuration > 1;
}
void AnalogClock::setAnimation(uint16_t duration, EasingEquation animationProgressionEquation)
{
animationDuration = duration;
animationEquation = animationProgressionEquation;
}
} // namespace touchgfx

View File

@ -0,0 +1,107 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <touchgfx/containers/clock/DigitalClock.hpp>
namespace touchgfx
{
DigitalClock::DigitalClock()
: AbstractClock(),
displayMode(DISPLAY_24_HOUR),
useLeadingZeroForHourIndicator(false)
{
buffer[0] = '\0';
text.setXY(0, 0);
text.setWildcard(buffer);
Container::add(text);
}
void DigitalClock::setWidth(int16_t width)
{
Container::setWidth(width);
text.setWidth(width);
}
void DigitalClock::setHeight(int16_t height)
{
Container::setHeight(height);
text.setHeight(height);
}
void DigitalClock::setBaselineY(int16_t baselineY)
{
if (text.getTypedText().hasValidId())
{
moveTo(getX(), baselineY - text.getTypedText().getFont()->getFontHeight());
}
}
void DigitalClock::displayLeadingZeroForHourIndicator(bool displayLeadingZero)
{
useLeadingZeroForHourIndicator = displayLeadingZero;
}
void DigitalClock::setAlpha(uint8_t newAlpha)
{
text.setAlpha(newAlpha);
}
uint8_t DigitalClock::getAlpha() const
{
return text.getAlpha();
}
void DigitalClock::setTypedText(TypedText typedText)
{
text.setTypedText(typedText);
text.invalidate();
}
void DigitalClock::setColor(colortype color)
{
text.setColor(color);
text.invalidate();
}
colortype DigitalClock::getColor() const
{
return text.getColor();
}
void DigitalClock::updateClock()
{
if (displayMode == DISPLAY_12_HOUR_NO_SECONDS)
{
const char* format = useLeadingZeroForHourIndicator ? "%02d:%02d %cM" : "%d:%02d %cM";
Unicode::snprintf(buffer, BUFFER_SIZE, format, getCurrentHour12(), getCurrentMinute(), getCurrentAM() ? 'A' : 'P');
}
else if (displayMode == DISPLAY_24_HOUR_NO_SECONDS)
{
const char* format = useLeadingZeroForHourIndicator ? "%02d:%02d" : "%d:%02d";
Unicode::snprintf(buffer, BUFFER_SIZE, format, getCurrentHour24(), getCurrentMinute());
}
else if (displayMode == DISPLAY_12_HOUR)
{
const char* format = useLeadingZeroForHourIndicator ? "%02d:%02d:%02d %cM" : "%d:%02d:%02d %cM";
Unicode::snprintf(buffer, BUFFER_SIZE, format, getCurrentHour12(), getCurrentMinute(), getCurrentSecond(), getCurrentAM() ? 'A' : 'P');
}
else if (displayMode == DISPLAY_24_HOUR)
{
const char* format = useLeadingZeroForHourIndicator ? "%02d:%02d:%02d" : "%d:%02d:%02d";
Unicode::snprintf(buffer, BUFFER_SIZE, format, getCurrentHour24(), getCurrentMinute(), getCurrentSecond());
}
text.invalidate();
}
} // namespace touchgfx

View File

@ -0,0 +1,40 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <touchgfx/containers/progress_indicators/AbstractDirectionProgress.hpp>
namespace touchgfx
{
AbstractDirectionProgress::AbstractDirectionProgress()
: AbstractProgressIndicator()
{
AbstractDirectionProgress::setDirection(RIGHT);
}
void AbstractDirectionProgress::setDirection(DirectionType direction)
{
if (direction != progressDirection)
{
progressDirection = direction;
progressIndicatorContainer.invalidate();
setValue(getValue());
}
}
AbstractDirectionProgress::DirectionType AbstractDirectionProgress::getDirection() const
{
return progressDirection;
}
} // namespace touchgfx

View File

@ -0,0 +1,224 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <touchgfx/containers/progress_indicators/AbstractProgressIndicator.hpp>
namespace touchgfx
{
AbstractProgressIndicator::AbstractProgressIndicator()
: Container(), rangeMin(0), rangeMax(100), currentValue(0), rangeSteps(100), rangeStepsMin(0),
equation(&EasingEquations::linearEaseNone), animationStartValue(0), animationEndValue(0), animationDuration(0), animationStep(0),
valueSetCallback(0), valueUpdatedCallback(0)
{
background.setXY(0, 0);
AbstractProgressIndicator::add(background);
AbstractProgressIndicator::add(progressIndicatorContainer);
}
void AbstractProgressIndicator::setBackground(const Bitmap& bitmapBackground)
{
background.setBitmap(bitmapBackground);
setWidthHeight(background);
}
void AbstractProgressIndicator::setProgressIndicatorPosition(int16_t x, int16_t y, int16_t width, int16_t height)
{
progressIndicatorContainer.setPosition(x, y, width, height);
if (getWidth() < x + width)
{
AbstractProgressIndicator::setWidth(x + width);
}
if (getHeight() < y + height)
{
AbstractProgressIndicator::setHeight(y + height);
}
}
int16_t AbstractProgressIndicator::getProgressIndicatorX() const
{
return progressIndicatorContainer.getX();
}
int16_t AbstractProgressIndicator::getProgressIndicatorY() const
{
return progressIndicatorContainer.getY();
}
int16_t AbstractProgressIndicator::getProgressIndicatorWidth() const
{
return progressIndicatorContainer.getWidth();
}
int16_t AbstractProgressIndicator::getProgressIndicatorHeight() const
{
return progressIndicatorContainer.getHeight();
}
void AbstractProgressIndicator::setRange(int min, int max, uint16_t steps /*= 0*/, uint16_t minStep /*= 0*/)
{
assert(min < max);
rangeMin = min;
rangeMax = max;
setValue(currentValue);
if (steps == 0)
{
rangeSteps = max - min;
}
else
{
rangeSteps = steps;
}
rangeStepsMin = minStep;
assert(rangeStepsMin < rangeSteps);
}
void AbstractProgressIndicator::getRange(int& min, int& max, uint16_t& steps, uint16_t& minStep) const
{
min = rangeMin;
max = rangeMax;
steps = rangeSteps;
minStep = rangeStepsMin;
}
void AbstractProgressIndicator::getRange(int& min, int& max, uint16_t& steps) const
{
min = rangeMin;
max = rangeMax;
steps = rangeSteps;
}
void AbstractProgressIndicator::getRange(int& min, int& max) const
{
min = rangeMin;
max = rangeMax;
}
void AbstractProgressIndicator::setValue(int value)
{
value = MAX(value, rangeMin);
value = MIN(value, rangeMax);
if (value != currentValue)
{
currentValue = value;
if (valueSetCallback && valueSetCallback->isValid())
{
valueSetCallback->execute(*this);
}
}
}
void AbstractProgressIndicator::setEasingEquation(EasingEquation easingEquation)
{
equation = easingEquation;
}
void AbstractProgressIndicator::updateValue(int value, uint16_t duration)
{
value = MAX(value, rangeMin);
value = MIN(value, rangeMax);
if (duration == 0)
{
setValue(value);
if (valueUpdatedCallback && valueUpdatedCallback->isValid())
{
valueUpdatedCallback->execute(*this);
}
return;
}
if (animationDuration > 0)
{
// Old animation is running, stop it first
Application::getInstance()->unregisterTimerWidget(this);
}
animationStartValue = getValue();
animationEndValue = value;
animationDuration = duration;
animationStep = 0;
Application::getInstance()->registerTimerWidget(this);
}
int AbstractProgressIndicator::getValue() const
{
return currentValue;
}
uint16_t AbstractProgressIndicator::getProgress(uint16_t range /*= 100*/) const
{
if (range == 0)
{
return 0;
}
int32_t remainder; // Not used here
// Find out at what step the current value is.
int32_t step = rangeStepsMin + muldiv(currentValue - rangeMin, rangeSteps - rangeStepsMin, rangeMax - rangeMin, remainder);
// Scale the step up to [0..range]
int32_t prog = muldiv(step, range, rangeSteps, remainder);
return (uint16_t)prog;
}
void AbstractProgressIndicator::setValueSetAction(GenericCallback<const AbstractProgressIndicator&>& callback)
{
valueSetCallback = &callback;
}
void AbstractProgressIndicator::handleTickEvent()
{
animationStep++;
int16_t delta = (int16_t)equation(animationStep, 0, animationEndValue - animationStartValue, animationDuration);
setValue(animationStartValue + delta);
if (animationStep >= animationDuration)
{
animationDuration = 0;
animationStep = 0;
Application::getInstance()->unregisterTimerWidget(this);
if (valueUpdatedCallback && valueUpdatedCallback->isValid())
{
valueUpdatedCallback->execute(*this);
}
}
}
void AbstractProgressIndicator::setValueUpdatedAction(GenericCallback<const AbstractProgressIndicator&>& callback)
{
valueUpdatedCallback = &callback;
}
void AbstractProgressIndicator::getRange(int16_t& min, int16_t& max, uint16_t& steps, uint16_t& minStep) const
{
int imin, imax;
getRange(imin, imax, steps, minStep);
min = (int16_t)imin;
max = (int16_t)imax;
}
void AbstractProgressIndicator::getRange(int16_t& min, int16_t& max, uint16_t& steps) const
{
int imin, imax;
getRange(imin, imax, steps);
min = (int16_t)imin;
max = (int16_t)imax;
}
void AbstractProgressIndicator::getRange(int16_t& min, int16_t& max) const
{
int imin, imax;
getRange(imin, imax);
min = (int16_t)imin;
max = (int16_t)imax;
}
} // namespace touchgfx

View File

@ -0,0 +1,108 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <touchgfx/containers/progress_indicators/BoxProgress.hpp>
namespace touchgfx
{
BoxProgress::BoxProgress()
: AbstractDirectionProgress(), box()
{
progressIndicatorContainer.add(box);
}
void BoxProgress::setProgressIndicatorPosition(int16_t x, int16_t y, int16_t width, int16_t height)
{
box.setPosition(0, 0, width, height);
AbstractProgressIndicator::setProgressIndicatorPosition(x, y, width, height);
}
void BoxProgress::setColor(colortype color)
{
box.setColor(color);
}
colortype BoxProgress::getColor() const
{
return box.getColor();
}
void BoxProgress::setAlpha(uint8_t newAlpha)
{
box.setAlpha(newAlpha);
}
uint8_t BoxProgress::getAlpha() const
{
return box.getAlpha();
}
void BoxProgress::setValue(int value)
{
AbstractProgressIndicator::setValue(value);
int16_t progress = 0;
switch (progressDirection)
{
case RIGHT:
case LEFT:
progress = AbstractProgressIndicator::getProgress(progressIndicatorContainer.getWidth());
break;
case DOWN:
case UP:
progress = AbstractProgressIndicator::getProgress(progressIndicatorContainer.getHeight());
break;
}
switch (progressDirection)
{
case RIGHT:
{
int16_t oldWidth = box.getWidth();
box.setPosition(0, 0, progress, progressIndicatorContainer.getHeight());
int16_t newWidth = box.getWidth();
Rect rect(MIN(oldWidth, newWidth), 0, abs(oldWidth - newWidth), box.getHeight());
progressIndicatorContainer.invalidateRect(rect);
break;
}
case LEFT:
{
int16_t oldX = box.getX();
box.setPosition(getWidth() - progress, 0, progress, progressIndicatorContainer.getHeight());
int16_t newX = box.getX();
Rect rect(MIN(oldX, newX), 0, abs(oldX - newX), box.getHeight());
progressIndicatorContainer.invalidateRect(rect);
break;
}
case DOWN:
{
int16_t oldHeight = box.getHeight();
box.setPosition(0, 0, progressIndicatorContainer.getWidth(), progress);
int16_t newHeight = box.getHeight();
Rect rect(0, MIN(oldHeight, newHeight), box.getWidth(), abs(oldHeight - newHeight));
progressIndicatorContainer.invalidateRect(rect);
break;
}
case UP:
{
int16_t oldY = box.getY();
box.setPosition(0, progressIndicatorContainer.getHeight() - progress, progressIndicatorContainer.getWidth(), progress);
int16_t newY = box.getY();
Rect rect(0, MIN(oldY, newY), box.getWidth(), abs(oldY - newY));
progressIndicatorContainer.invalidateRect(rect);
break;
}
}
}
} // namespace touchgfx

View File

@ -0,0 +1,124 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <touchgfx/containers/progress_indicators/CircleProgress.hpp>
namespace touchgfx
{
CircleProgress::CircleProgress()
: AbstractProgressIndicator(), circle()
{
progressIndicatorContainer.add(circle);
circle.setPosition(0, 0, getWidth(), getHeight());
CircleProgress::setStartEndAngle(0, 360);
}
void CircleProgress::setProgressIndicatorPosition(int16_t x, int16_t y, int16_t width, int16_t height)
{
circle.setPosition(0, 0, width, height);
AbstractProgressIndicator::setProgressIndicatorPosition(x, y, width, height);
}
void CircleProgress::setPainter(AbstractPainter& painter)
{
circle.setPainter(painter);
}
void CircleProgress::setCenter(int x, int y)
{
circle.setCenter(x, y);
}
void CircleProgress::getCenter(int& x, int& y) const
{
circle.getCenter(x, y);
}
void CircleProgress::setRadius(int r)
{
circle.setRadius(r);
}
int CircleProgress::getRadius() const
{
int radius;
circle.getRadius(radius);
return radius;
}
void CircleProgress::setLineWidth(int width)
{
circle.setLineWidth(width);
}
int CircleProgress::getLineWidth() const
{
int width;
circle.getLineWidth(width);
return width;
}
void CircleProgress::setCapPrecision(int precision)
{
circle.setCapPrecision(precision);
}
void CircleProgress::setStartEndAngle(int startAngle, int endAngle)
{
assert(startAngle != endAngle);
circle.setArc(startAngle, endAngle);
circleEndAngle = endAngle;
CircleProgress::setValue(CircleProgress::getValue());
}
int CircleProgress::getStartAngle() const
{
return circle.getArcStart();
}
int CircleProgress::getEndAngle() const
{
return circleEndAngle;
}
void CircleProgress::setAlpha(uint8_t newAlpha)
{
circle.setAlpha(newAlpha);
}
uint8_t CircleProgress::getAlpha() const
{
return circle.getAlpha();
}
void CircleProgress::setValue(int value)
{
AbstractProgressIndicator::setValue(value);
CWRUtil::Q5 startAngle;
CWRUtil::Q5 endAngle = CWRUtil::toQ5(circleEndAngle);
circle.getArcStart<CWRUtil::Q5>(startAngle);
uint16_t rangeAngleSteps = endAngle < startAngle ? (int)(startAngle - endAngle) : (int)(endAngle - startAngle);
CWRUtil::Q5 progress = CWRUtil::Q5(AbstractProgressIndicator::getProgress(rangeAngleSteps));
if (endAngle < startAngle)
{
circle.updateArcEnd<CWRUtil::Q5>(startAngle - progress);
}
else
{
circle.updateArcEnd<CWRUtil::Q5>(startAngle + progress);
}
}
} // namespace touchgfx

View File

@ -0,0 +1,141 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <touchgfx/containers/progress_indicators/ImageProgress.hpp>
namespace touchgfx
{
ImageProgress::ImageProgress()
: AbstractDirectionProgress(), image(), fixedPosition(true)
{
container.add(image);
progressIndicatorContainer.add(container);
}
void ImageProgress::setProgressIndicatorPosition(int16_t x, int16_t y, int16_t width, int16_t height)
{
container.setPosition(0, 0, width, height);
AbstractProgressIndicator::setProgressIndicatorPosition(x, y, width, height);
}
void ImageProgress::setAnchorAtZero(bool anchorAtZero)
{
fixedPosition = anchorAtZero;
setValue(getValue());
}
bool ImageProgress::getAnchorAtZero() const
{
return fixedPosition;
}
void ImageProgress::setBitmap(BitmapId bitmapId)
{
image.setBitmap(Bitmap(bitmapId));
}
BitmapId ImageProgress::getBitmap() const
{
return image.getBitmapId();
}
void ImageProgress::setAlpha(uint8_t newAlpha)
{
image.setAlpha(newAlpha);
}
uint8_t ImageProgress::getAlpha() const
{
return image.getAlpha();
}
void ImageProgress::setValue(int value)
{
AbstractProgressIndicator::setValue(value);
const uint16_t maxProgress = (progressDirection == RIGHT || progressDirection == LEFT) ? progressIndicatorContainer.getWidth() : progressIndicatorContainer.getHeight();
int16_t progress = AbstractProgressIndicator::getProgress(maxProgress);
if (fixedPosition)
{
switch (progressDirection)
{
case RIGHT:
{
int16_t oldWidth = container.getWidth();
container.setPosition(0, 0, progress, progressIndicatorContainer.getHeight());
image.setPosition(0, 0, progress, progressIndicatorContainer.getHeight());
int16_t newWidth = container.getWidth();
Rect rect(MIN(oldWidth, newWidth), 0, abs(oldWidth - newWidth), container.getHeight());
progressIndicatorContainer.invalidateRect(rect);
break;
}
case LEFT:
{
int16_t oldX = container.getX();
container.setPosition(getWidth() - progress, 0, progress, progressIndicatorContainer.getHeight());
image.setPosition(-container.getX(), 0, progressIndicatorContainer.getWidth(), progressIndicatorContainer.getHeight());
int16_t newX = container.getX();
Rect rect(MIN(oldX, newX), 0, abs(oldX - newX), container.getHeight());
progressIndicatorContainer.invalidateRect(rect);
break;
}
case DOWN:
{
int16_t oldHeight = container.getHeight();
container.setPosition(0, 0, progressIndicatorContainer.getWidth(), progress);
image.setPosition(0, 0, progressIndicatorContainer.getWidth(), progress);
int16_t newHeight = container.getHeight();
Rect rect(0, MIN(oldHeight, newHeight), container.getWidth(), abs(oldHeight - newHeight));
progressIndicatorContainer.invalidateRect(rect);
break;
}
case UP:
{
int16_t oldY = container.getY();
container.setPosition(0, progressIndicatorContainer.getHeight() - progress, progressIndicatorContainer.getWidth(), progress);
image.setPosition(0, -container.getY(), progressIndicatorContainer.getWidth(), progressIndicatorContainer.getHeight());
int16_t newY = container.getY();
Rect rect(0, MIN(oldY, newY), container.getWidth(), abs(oldY - newY));
progressIndicatorContainer.invalidateRect(rect);
break;
}
}
}
else
{
container.invalidate();
switch (progressDirection)
{
case RIGHT:
container.setPosition(0, 0, progress, getHeight());
image.setPosition(progress - progressIndicatorContainer.getWidth(), 0, progressIndicatorContainer.getWidth(), getHeight());
break;
case LEFT:
container.setPosition(progressIndicatorContainer.getWidth() - progress, 0, progress, progressIndicatorContainer.getHeight());
image.setPosition(0, 0, progress, progressIndicatorContainer.getHeight());
break;
case DOWN:
container.setPosition(0, 0, progressIndicatorContainer.getWidth(), progress);
image.setPosition(0, progress - progressIndicatorContainer.getHeight(), progressIndicatorContainer.getWidth(), progressIndicatorContainer.getHeight());
break;
case UP:
container.setPosition(0, progressIndicatorContainer.getHeight() - progress, progressIndicatorContainer.getWidth(), progress);
image.setPosition(0, 0, progressIndicatorContainer.getWidth(), progress);
break;
}
container.invalidate();
}
}
} // namespace touchgfx

View File

@ -0,0 +1,109 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <touchgfx/containers/progress_indicators/LineProgress.hpp>
namespace touchgfx
{
LineProgress::LineProgress()
: AbstractProgressIndicator(), line(), endX(0), endY(0)
{
progressIndicatorContainer.add(line);
line.setPosition(0, 0, getWidth(), getHeight());
}
void LineProgress::setProgressIndicatorPosition(int16_t x, int16_t y, int16_t width, int16_t height)
{
line.setPosition(0, 0, width, height);
AbstractProgressIndicator::setProgressIndicatorPosition(x, y, width, height);
}
void LineProgress::setPainter(AbstractPainter& painter)
{
line.setPainter(painter);
}
void LineProgress::setStart(int x, int y)
{
startX = CWRUtil::toQ5<int>(x);
startY = CWRUtil::toQ5<int>(y);
line.setStart(x, y);
}
void LineProgress::getStart(int& x, int& y) const
{
x = startX.to<int>();
y = startY.to<int>();
}
void LineProgress::setEnd(int x, int y)
{
endX = CWRUtil::toQ5<int>(x);
endY = CWRUtil::toQ5<int>(y);
}
void LineProgress::getEnd(int& x, int& y) const
{
x = endX.to<int>();
y = endY.to<int>();
}
void LineProgress::setLineWidth(int width)
{
line.setLineWidth(width);
}
int LineProgress::getLineWidth() const
{
int width;
line.getLineWidth(width);
return width;
}
void LineProgress::setLineEndingStyle(Line::LINE_ENDING_STYLE lineEndingStyle)
{
line.setLineEndingStyle(lineEndingStyle);
}
Line::LINE_ENDING_STYLE LineProgress::getLineEndingStyle() const
{
return line.getLineEndingStyle();
}
void LineProgress::setAlpha(uint8_t newAlpha)
{
line.setAlpha(newAlpha);
}
uint8_t LineProgress::getAlpha() const
{
return line.getAlpha();
}
void LineProgress::setValue(int value)
{
if (rangeSteps > 0)
{
AbstractProgressIndicator::setValue(value);
int progress = (int)AbstractProgressIndicator::getProgress(rangeSteps);
CWRUtil::Q5 r(rangeSteps);
CWRUtil::Q5 p(progress);
CWRUtil::Q5 x = startX + (endX - startX) / r * p;
CWRUtil::Q5 y = startY + (endY - startY) / r * p;
line.updateEnd(x, y);
}
}
} // namespace touchgfx

View File

@ -0,0 +1,93 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <touchgfx/containers/progress_indicators/TextProgress.hpp>
namespace touchgfx
{
TextProgress::TextProgress()
: AbstractProgressIndicator(),
textArea(),
decimals(0)
{
textBuffer[0] = 0;
progressIndicatorContainer.add(textArea);
}
void TextProgress::setProgressIndicatorPosition(int16_t x, int16_t y, int16_t width, int16_t height)
{
textArea.setPosition(0, 0, width, height);
AbstractProgressIndicator::setProgressIndicatorPosition(x, y, width, height);
}
void TextProgress::setTypedText(const TypedText& t)
{
textArea.setTypedText(t);
}
TypedText TextProgress::getTypedText() const
{
return textArea.getTypedText();
}
void TextProgress::setColor(colortype color)
{
textArea.setColor(color);
}
colortype TextProgress::getColor() const
{
return textArea.getColor();
}
void TextProgress::setAlpha(uint8_t newAlpha)
{
textArea.setAlpha(newAlpha);
}
uint8_t TextProgress::getAlpha() const
{
return textArea.getAlpha();
}
void TextProgress::setValue(int value)
{
AbstractProgressIndicator::setValue(value);
int range[3] = { 1, 10, 100 };
uint16_t progress = AbstractProgressIndicator::getProgress(100 * range[decimals]);
if (decimals > 0)
{
Unicode::snprintf(textBuffer, 8, "%d.%0*d", progress / range[decimals], decimals, progress % range[decimals]);
}
else
{
Unicode::snprintf(textBuffer, 8, "%d", progress);
}
textArea.setWildcard(textBuffer);
textArea.invalidate();
}
void TextProgress::setNumberOfDecimals(uint16_t numberOfDecimals)
{
decimals = MIN(2, numberOfDecimals);
setValue(getValue());
}
uint16_t TextProgress::getNumberOfDecimals() const
{
return decimals;
}
} // namespace touchgfx

View File

@ -0,0 +1,385 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <touchgfx/containers/scrollers/DrawableList.hpp>
namespace touchgfx
{
DrawableList::DrawableList()
: Container(),
isHorizontal(false),
isCircular(false),
offset(0),
itemSize(0),
itemMargin(0),
numItems(0),
numDrawables(0),
firstItem(0),
firstDrawable(0),
drawablesInitialized(false),
firstDrawableIndex(0),
drawableItems(0),
updateDrawable(0)
{
}
void DrawableList::setWidth(int16_t width)
{
Container::setWidth(width);
refreshDrawables();
}
void DrawableList::setHeight(int16_t height)
{
Container::setHeight(height);
refreshDrawables();
}
void DrawableList::setHorizontal(bool horizontal)
{
if ((horizontal && !isHorizontal) || (!horizontal && isHorizontal))
{
isHorizontal = horizontal;
refreshDrawables();
}
}
bool DrawableList::getHorizontal() const
{
return isHorizontal;
}
void DrawableList::setCircular(bool circular)
{
if ((circular && !isCircular) || (!circular && isCircular))
{
isCircular = circular;
refreshDrawables();
}
}
bool DrawableList::getCircular() const
{
return isCircular;
}
void DrawableList::setDrawableSize(int16_t drawableSize, int16_t drawableMargin)
{
itemSize = drawableSize + 2 * drawableMargin;
itemMargin = drawableMargin;
}
void DrawableList::setDrawables(DrawableListItemsInterface& drawableListItems,
int16_t drawableItemIndexOffset,
GenericCallback<DrawableListItemsInterface*, int16_t, int16_t>& updateDrawableCallback)
{
drawableItems = &drawableListItems;
firstDrawableIndex = drawableItemIndexOffset;
updateDrawable = &updateDrawableCallback;
refreshDrawables();
}
int16_t DrawableList::getNumberOfDrawables() const
{
return numDrawables;
}
int16_t DrawableList::getItemSize() const
{
return itemSize;
}
int16_t DrawableList::getDrawableSize() const
{
return itemSize - 2 * itemMargin;
}
int16_t DrawableList::getDrawableMargin() const
{
return itemMargin;
}
void DrawableList::setNumberOfItems(int16_t numberOfItems)
{
numItems = numberOfItems;
refreshDrawables();
}
int16_t DrawableList::getNumberOfItems() const
{
return numItems;
}
int16_t DrawableList::getRequiredNumberOfDrawables() const
{
if (drawableItems == 0 || itemSize <= 0)
{
return 0;
}
// Calculate number of required drawables. Worst case is one pixel visible of drawable at top and rest stacked tightly
int16_t requiredDrawables = 1 + (((isHorizontal ? getWidth() : getHeight()) - 1) + (itemSize - 1)) / itemSize;
if (!isCircular)
{
// We never require more drawables than the number of elements on non-circular list.
if (requiredDrawables > numItems)
{
requiredDrawables = numItems;
}
}
int16_t numberOfDrawables = drawableItems->getNumberOfDrawables();
return MIN((numberOfDrawables - firstDrawableIndex), requiredDrawables);
}
void DrawableList::setOffset(int32_t ofs)
{
offset = ofs;
if (numDrawables == 0 || numItems == 0 || itemSize == 0)
{
return;
}
if (!updateDrawable || !updateDrawable->isValid())
{
return;
}
// ofs is the offset of item[0]
// 0 => item[0] is perfectly selected, -itemSize => item[1] is perfectly selected, itemSize => item[N-1] is perfectly selected etc.
int16_t newFirstItem = 0;
if (ofs > 0)
{
int numberOfItems = ofs / itemSize + 1;
newFirstItem -= numberOfItems;
ofs -= numberOfItems * itemSize;
}
if (ofs <= -itemSize)
{
int numberOfItems = ofs / itemSize;
newFirstItem -= numberOfItems;
ofs -= numberOfItems * itemSize;
}
if (isCircular)
{
// Make sure that firstIndex is "in range"
newFirstItem %= numItems;
newFirstItem = (newFirstItem + numItems) % numItems;
}
else
{
if (newFirstItem < 0)
{
ofs -= newFirstItem * itemSize;
newFirstItem = 0;
}
else if (newFirstItem + numDrawables > numItems)
{
int x = numItems - (newFirstItem + numDrawables);
ofs += x * itemSize;
newFirstItem += x;
}
}
int drawableDelta = 0;
if (drawablesInitialized && firstItem != newFirstItem)
{
drawableDelta = numDrawables;
for (int i = 1; i < numDrawables; i++)
{
int fi = (firstItem + i);
int nfi = (newFirstItem + i);
if (isCircular)
{
fi %= numItems;
nfi %= numItems;
}
if (fi == newFirstItem)
{
drawableDelta = -i;
break;
}
if (nfi == firstItem)
{
drawableDelta = i;
break;
}
}
}
firstDrawable = ((firstDrawable - drawableDelta) + numDrawables) % numDrawables;
firstItem = newFirstItem;
for (int i = 0; i < numDrawables; i++)
{
int drawableIndex = (firstDrawable + i) % numDrawables;
Drawable* drawable = drawableItems->getDrawable(drawableIndex + firstDrawableIndex);
if (isHorizontal)
{
drawable->moveTo(ofs + i * itemSize + itemMargin, 0);
}
else
{
drawable->moveTo(0, ofs + i * itemSize + itemMargin);
}
int itemIndex = i + firstItem;
if (isCircular)
{
itemIndex %= numItems;
}
else
{
if (itemIndex < 0 || itemIndex >= numItems)
{
itemIndex = -1;
}
}
if (itemIndex < 0)
{
drawable->setVisible(false);
}
else
{
drawable->setVisible(true);
// Only fill if first time or outside old range
if (!drawablesInitialized || (i < drawableDelta || i >= numDrawables + drawableDelta))
{
if (updateDrawable->isValid())
{
updateDrawable->execute(drawableItems, drawableIndex + firstDrawableIndex, itemIndex);
drawable->invalidate();
}
}
}
}
drawablesInitialized = true;
}
int32_t DrawableList::getOffset() const
{
return offset;
}
int16_t DrawableList::getItemIndex(int16_t drawableIndex) const
{
if (drawableIndex < 0 || drawableIndex >= numDrawables || numDrawables == 0 || numItems == 0)
{
return -1;
}
int16_t itemNumber = ((drawableIndex + numDrawables - firstDrawable) % numDrawables) + firstItem;
if (isCircular)
{
itemNumber %= numItems;
}
if (itemNumber < 0 || itemNumber >= numItems)
{
return -1;
}
return itemNumber;
}
int16_t DrawableList::getDrawableIndices(int16_t itemIndex, int16_t* drawableIndexArray, int16_t arraySize) const
{
int16_t numFound = 0;
int16_t drawableIndex = -1;
while ((drawableIndex = getDrawableIndex(itemIndex, drawableIndex)) >= 0)
{
if (numFound < arraySize)
{
drawableIndexArray[numFound] = drawableIndex;
numFound++;
}
}
return numFound;
}
int16_t DrawableList::getDrawableIndex(int16_t itemIndex, int16_t prevDrawableIndex /*= -1*/) const
{
if (prevDrawableIndex < -1 || prevDrawableIndex >= numDrawables || numDrawables == 0 || numItems == 0)
{
return -1;
}
if (prevDrawableIndex >= 0)
{
prevDrawableIndex = (prevDrawableIndex - firstDrawable + numDrawables) % numDrawables;
}
for (int16_t i = prevDrawableIndex + 1; i < numDrawables; i++)
{
int16_t currentItemIndex = firstItem + i;
if (isCircular)
{
currentItemIndex %= numItems;
}
if (itemIndex == currentItemIndex)
{
int16_t drawableIndex = (firstDrawable + i) % numDrawables;
return drawableIndex;
}
}
return -1;
}
void DrawableList::refreshDrawables()
{
if (drawableItems == 0)
{
numDrawables = 0;
return;
}
numDrawables = getRequiredNumberOfDrawables();
// Remove everything
Container::removeAll();
// Add the itemDrawables
for (int drawableIndex = 0; drawableIndex < numDrawables; drawableIndex++)
{
Drawable* drawable = drawableItems->getDrawable(drawableIndex + firstDrawableIndex);
// Resize the drawables, X/Y ignored for now.
if (isHorizontal)
{
drawable->setPosition(0, 0, itemSize - 2 * itemMargin, getHeight());
}
else
{
drawable->setPosition(0, 0, getWidth(), itemSize - 2 * itemMargin);
}
// Add each drawable for later positioning
if (drawable->getParent() != 0)
{
// Remove drawable from the current parent
Container* parent = static_cast<Container*>(drawable->getParent());
parent->remove(*drawable);
}
Container::add(*drawable);
}
drawablesInitialized = false;
firstItem = 0;
firstDrawable = 0;
setOffset(offset);
}
void DrawableList::itemChanged(int16_t itemIndex)
{
if (updateDrawable && updateDrawable->isValid())
{
int16_t drawableIndex = -1;
while ((drawableIndex = getDrawableIndex(itemIndex, drawableIndex)) != -1)
{
updateDrawable->execute(drawableItems, drawableIndex + firstDrawableIndex, itemIndex);
}
}
}
} // namespace touchgfx

View File

@ -0,0 +1,336 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <touchgfx/containers/scrollers/ScrollBase.hpp>
namespace touchgfx
{
ScrollBase::ScrollBase()
: Container(),
numberOfDrawables(0),
distanceBeforeAlignedItem(0),
itemSize(0),
swipeAcceleration(10),
dragAcceleration(10),
maxSwipeItems(0),
easingEquation(&EasingEquations::backEaseOut),
defaultAnimationSteps(30),
itemSelectedCallback(0),
itemLockedInCallback(0),
animationEndedCallback(0),
itemPressedCallback(0),
currentAnimationState(NO_ANIMATION),
gestureStep(0),
gestureStepsTotal(0),
gestureStart(0),
gestureEnd(0),
xClick(0),
yClick(0),
initialSwipeOffset(0),
draggableX(false),
draggableY(false)
{
Container::add(list);
list.setXY(0, 0);
list.setHorizontal(false);
list.setCircular(false);
setTouchable(true);
}
void ScrollBase::setWidth(int16_t width)
{
Container::setWidth(width);
list.setWidth(width);
}
void ScrollBase::setHeight(int16_t height)
{
Container::setHeight(height);
list.setHeight(height);
}
void ScrollBase::setHorizontal(bool horizontal)
{
allowVerticalDrag(horizontal);
allowHorizontalDrag(!horizontal);
list.setHorizontal(horizontal);
}
bool ScrollBase::getHorizontal() const
{
return list.getHorizontal();
}
void ScrollBase::setCircular(bool circular)
{
list.setCircular(circular);
}
bool ScrollBase::getCircular() const
{
return list.getCircular();
}
void ScrollBase::setDrawableSize(int16_t drawableSize, int16_t drawableMargin)
{
itemSize = drawableSize + drawableMargin * 2;
list.setDrawableSize(drawableSize, drawableMargin);
}
int16_t ScrollBase::getDrawableSize() const
{
return list.getDrawableSize();
}
int16_t ScrollBase::getDrawableMargin() const
{
return list.getDrawableMargin();
}
void ScrollBase::setNumberOfItems(int16_t numberOfItems)
{
if (numberOfItems != getNumberOfItems())
{
list.setNumberOfItems(numberOfItems);
if (!getCircular())
{
animateToPosition(keepOffsetInsideLimits(getOffset(), 0));
}
}
}
int16_t ScrollBase::getNumberOfItems() const
{
return list.getNumberOfItems();
}
void ScrollBase::setEasingEquation(EasingEquation equation)
{
easingEquation = equation;
}
void ScrollBase::setAnimationSteps(int16_t steps)
{
defaultAnimationSteps = steps;
}
uint16_t ScrollBase::getAnimationSteps() const
{
return defaultAnimationSteps;
}
void ScrollBase::setSwipeAcceleration(uint16_t acceleration)
{
swipeAcceleration = acceleration;
}
uint16_t ScrollBase::getSwipeAcceleration() const
{
return swipeAcceleration;
}
void ScrollBase::setMaxSwipeItems(uint16_t maxItems)
{
maxSwipeItems = maxItems;
}
uint16_t ScrollBase::getMaxSwipeItems() const
{
return maxSwipeItems;
}
void ScrollBase::setDragAcceleration(uint16_t acceleration)
{
dragAcceleration = acceleration;
}
uint16_t ScrollBase::getDragAcceleration() const
{
return dragAcceleration;
}
void ScrollBase::allowHorizontalDrag(bool enable)
{
draggableX = enable;
}
void ScrollBase::allowVerticalDrag(bool enable)
{
draggableY = enable;
}
void ScrollBase::animateToItem(int16_t itemIndex, int16_t animationSteps /*= -1*/)
{
int32_t position = getPositionForItem(itemIndex);
animateToPosition(position, animationSteps);
}
void ScrollBase::setItemSelectedCallback(GenericCallback<int16_t>& callback)
{
itemSelectedCallback = &callback;
}
void ScrollBase::setAnimationEndedCallback(GenericCallback<>& callback)
{
animationEndedCallback = &callback;
}
void ScrollBase::setItemPressedCallback(GenericCallback<int16_t>& callback)
{
itemPressedCallback = &callback;
}
bool ScrollBase::isAnimating() const
{
return currentAnimationState != NO_ANIMATION;
}
void ScrollBase::stopAnimation()
{
if (currentAnimationState == ANIMATING_GESTURE)
{
Application::getInstance()->unregisterTimerWidget(this);
setOffset(gestureEnd);
}
currentAnimationState = NO_ANIMATION;
}
void ScrollBase::handleDragEvent(const DragEvent& evt)
{
stopAnimation();
currentAnimationState = ANIMATING_DRAG;
int32_t newOffset = getOffset() + (getHorizontal() ? evt.getDeltaX() : evt.getDeltaY()) * dragAcceleration / 10;
newOffset = keepOffsetInsideLimits(newOffset, itemSize * 3 / 4);
setOffset(newOffset);
}
void ScrollBase::handleGestureEvent(const GestureEvent& evt)
{
if (evt.getType() == (getHorizontal() ? GestureEvent::SWIPE_HORIZONTAL : GestureEvent::SWIPE_VERTICAL))
{
int16_t velocity = abs(evt.getVelocity());
int16_t direction = evt.getVelocity() < 0 ? -1 : 1;
int16_t steps = MAX(1, velocity - 4) * 7;
int32_t newOffset = getOffset() + direction * steps * swipeAcceleration / 10;
if (maxSwipeItems > 0)
{
int32_t maxDistance = maxSwipeItems * itemSize;
newOffset = MIN(newOffset, initialSwipeOffset + maxDistance);
newOffset = MAX(newOffset, initialSwipeOffset - maxDistance);
}
newOffset = keepOffsetInsideLimits(newOffset, 0);
steps = MIN(steps, defaultAnimationSteps);
animateToPosition(newOffset, steps);
}
}
void ScrollBase::handleTickEvent()
{
if (currentAnimationState == ANIMATING_GESTURE)
{
gestureStep++;
int newPosition = gestureStart + easingEquation(gestureStep, 0, gestureEnd - gestureStart, gestureStepsTotal);
setOffset(newPosition);
if (gestureStep > gestureStepsTotal)
{
currentAnimationState = NO_ANIMATION;
gestureStep = 0;
Application::getInstance()->unregisterTimerWidget(this);
setOffset(getNormalizedOffset(gestureEnd));
// Also adjust initialSwipeOffset in case it is being used.
initialSwipeOffset += getOffset() - gestureEnd;
//Item has settled, call back
if (animationEndedCallback && animationEndedCallback->isValid())
{
animationEndedCallback->execute();
}
}
}
}
void ScrollBase::itemChanged(int itemIndex)
{
list.itemChanged(itemIndex);
}
void ScrollBase::setOffset(int32_t offset)
{
list.setOffset(offset + distanceBeforeAlignedItem);
}
int32_t ScrollBase::getOffset() const
{
return list.getOffset() - distanceBeforeAlignedItem;
}
int ScrollBase::getNormalizedOffset(int offset) const
{
int16_t numItems = getNumberOfItems();
if (numItems == 0 || itemSize == 0)
{
return offset;
}
int32_t listSize = numItems * itemSize;
offset %= listSize;
return offset > 0 ? offset - listSize : offset;
}
int32_t ScrollBase::getNearestAlignedOffset(int32_t offset) const
{
if (itemSize == 0)
{
return offset;
}
if (getCircular())
{
if (offset < 0)
{
return (((offset - (itemSize / 2)) / itemSize) * itemSize);
}
return ((offset + (itemSize / 2)) / itemSize) * itemSize;
}
offset = keepOffsetInsideLimits(offset, 0);
return ((offset - (itemSize / 2)) / itemSize) * itemSize;
}
void ScrollBase::animateToPosition(int32_t position, int16_t steps)
{
int32_t currentPosition = getOffset();
position = getNearestAlignedOffset(position);
if (steps < 0)
{
steps = defaultAnimationSteps;
}
steps = MIN(steps, abs(position - currentPosition));
if (steps < 1)
{
setOffset(position);
currentAnimationState = NO_ANIMATION;
}
else
{
gestureStart = currentPosition;
gestureEnd = position;
gestureStep = 0;
gestureStepsTotal = steps;
if (currentAnimationState != ANIMATING_GESTURE)
{
Application::getInstance()->registerTimerWidget(this);
currentAnimationState = ANIMATING_GESTURE;
}
}
}
} // namespace touchgfx

View File

@ -0,0 +1,230 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <touchgfx/containers/scrollers/ScrollList.hpp>
namespace touchgfx
{
ScrollList::ScrollList()
: ScrollBase(),
paddingAfterLastItem(0),
snapping(false),
windowSize(1)
{
}
void ScrollList::setDrawables(DrawableListItemsInterface& drawableListItems, GenericCallback<DrawableListItemsInterface*, int16_t, int16_t>& updateDrawableCallback)
{
stopAnimation();
numberOfDrawables = drawableListItems.getNumberOfDrawables();
list.setDrawables(drawableListItems, 0, updateDrawableCallback);
setOffset(0);
}
void ScrollList::setWindowSize(int16_t items)
{
windowSize = MAX(1, items);
animateToPosition(keepOffsetInsideLimits(getOffset(), 0));
}
void ScrollList::setPadding(int16_t paddingBefore, int16_t paddingAfter)
{
int32_t currentOffset = getOffset();
distanceBeforeAlignedItem = paddingBefore;
paddingAfterLastItem = paddingAfter;
setOffset(currentOffset);
list.refreshDrawables();
}
int16_t ScrollList::getPaddingBefore() const
{
return distanceBeforeAlignedItem;
}
int16_t ScrollList::getPaddingAfter() const
{
return paddingAfterLastItem;
}
void ScrollList::setSnapping(bool snap)
{
snapping = snap;
if (snapping)
{
setOffset(getNearestAlignedOffset(getOffset()));
}
}
bool ScrollList::getSnapping() const
{
return snapping;
}
int32_t ScrollList::getPositionForItem(int16_t itemIndex)
{
int32_t currentOffset = getNormalizedOffset(getOffset());
if (itemIndex < 0 || itemIndex >= list.getNumberOfItems() || itemSize == 0)
{
return currentOffset;
}
int32_t itemOffset = -itemIndex * itemSize;
// Get the visible size
int16_t widgetSize = getHorizontal() ? getWidth() : getHeight();
int16_t activeWidgetSize = widgetSize - (distanceBeforeAlignedItem + paddingAfterLastItem);
if (list.getCircular())
{
int32_t offset = currentOffset;
// Important this is a do-while of visibleSize < itemSize in which case we need to check at least one time
do
{
int16_t i = (-getNormalizedOffset(offset)) / itemSize; // Item index of first
if (itemIndex == i)
{
return currentOffset;
}
offset -= itemSize;
} while (offset >= currentOffset - (activeWidgetSize - itemSize));
int32_t allItemsSize = list.getNumberOfItems() * itemSize;
// Either scroll left from the first item or right from the last item. Find out which is closest
int32_t leftScrollDistance = itemOffset - currentOffset;
int32_t leftScrollDistance2 = (itemOffset + allItemsSize) - currentOffset;
int32_t rightItemOffset = getNormalizedOffset(currentOffset - (activeWidgetSize - itemSize));
int32_t rightScrollDistance = rightItemOffset - itemOffset;
int32_t rightScrollDistance2 = rightItemOffset - (itemOffset - allItemsSize);
if (abs(leftScrollDistance2) < abs(leftScrollDistance))
{
leftScrollDistance = leftScrollDistance2;
}
if (abs(rightScrollDistance2) < abs(rightScrollDistance))
{
rightScrollDistance = rightScrollDistance2;
}
if (abs(rightScrollDistance) < abs(leftScrollDistance))
{
return currentOffset - rightScrollDistance;
}
return currentOffset + leftScrollDistance;
}
else
{
if (itemOffset > currentOffset) // First item on screen is higher than the itemIndex. Scroll itemIndex to top position
{
return itemOffset;
}
int16_t numberOfVisibleItems = activeWidgetSize / itemSize;
int32_t itemOffsetAtEnd = itemOffset;
if (numberOfVisibleItems > 0)
{
if (snapping)
{
itemOffsetAtEnd = itemOffset + (numberOfVisibleItems - 1) * itemSize;
}
else
{
itemOffsetAtEnd = itemOffset + activeWidgetSize - itemSize;
}
}
if (itemOffsetAtEnd < currentOffset)
{
return itemOffsetAtEnd;
}
}
return currentOffset;
}
void ScrollList::handleClickEvent(const ClickEvent& evt)
{
ScrollBase::handleClickEvent(evt);
if (evt.getType() == ClickEvent::PRESSED)
{
xClick = evt.getX();
yClick = evt.getY();
initialSwipeOffset = getOffset();
setOffset(getNearestAlignedOffset(initialSwipeOffset));
if (itemPressedCallback && itemPressedCallback->isValid())
{
int16_t click = (getHorizontal() ? xClick : yClick);
int32_t offset = click - getOffset();
int32_t listSize = getNumberOfItems() * itemSize;
if (getCircular())
{
offset += listSize;
offset %= listSize;
}
if (offset >= 0 && offset < listSize)
{
int16_t item = offset / itemSize;
itemPressedCallback->execute(item);
}
}
}
else if (evt.getType() == ClickEvent::RELEASED)
{
if (currentAnimationState == NO_ANIMATION)
{
// For a tiny drag, start by re-aligning (no animation(!))
setOffset(getNearestAlignedOffset(getOffset()));
if (itemSelectedCallback && itemSelectedCallback->isValid())
{
int16_t click = (getHorizontal() ? xClick : yClick);
int32_t offset = click - getOffset();
int32_t listSize = getNumberOfItems() * itemSize;
if (getCircular())
{
offset += listSize;
offset %= listSize;
}
else
{
offset -= distanceBeforeAlignedItem;
}
if (offset >= 0 && offset < listSize)
{
int16_t item = offset / itemSize;
itemSelectedCallback->execute(item);
}
}
}
else if (currentAnimationState == ANIMATING_DRAG)
{
// click + drag + release. Find best Y to scroll to
animateToPosition(getNearestAlignedOffset(getOffset()));
}
}
}
int32_t ScrollList::getNearestAlignedOffset(int32_t offset) const
{
if (snapping)
{
// ScrollBase implementation will snap
return ScrollBase::getNearestAlignedOffset(offset);
}
return keepOffsetInsideLimits(offset, 0);
}
int32_t ScrollList::keepOffsetInsideLimits(int32_t newOffset, int16_t overShoot) const
{
if (!getCircular())
{
newOffset = MIN(newOffset, overShoot);
int maxOffToTheStart = windowSize < getNumberOfItems() ? getNumberOfItems() - windowSize : 0;
newOffset = MAX(newOffset, -(itemSize * maxOffToTheStart) - overShoot);
}
return newOffset;
}
} // namespace touchgfx

View File

@ -0,0 +1,29 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <touchgfx/containers/scrollers/ScrollWheel.hpp>
namespace touchgfx
{
void ScrollWheel::setDrawables(DrawableListItemsInterface& drawableListItems, GenericCallback<DrawableListItemsInterface*, int16_t, int16_t>& updateDrawableCallback)
{
stopAnimation();
numberOfDrawables = drawableListItems.getNumberOfDrawables();
list.setDrawables(drawableListItems, 0, updateDrawableCallback);
setOffset(0);
}
} // namespace touchgfx

View File

@ -0,0 +1,178 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <touchgfx/containers/scrollers/ScrollWheelBase.hpp>
namespace touchgfx
{
ScrollWheelBase::ScrollWheelBase()
: ScrollBase(),
animateToCallback(0)
{
ScrollWheelBase::setHorizontal(false);
setTouchable(true);
}
void ScrollWheelBase::setSelectedItemOffset(int16_t offset)
{
int32_t currentOffset = getOffset();
distanceBeforeAlignedItem = offset;
setOffset(currentOffset);
}
int16_t ScrollWheelBase::getSelectedItemOffset() const
{
return distanceBeforeAlignedItem;
}
int32_t ScrollWheelBase::getPositionForItem(int16_t itemIndex)
{
int32_t newOffset = -itemIndex * itemSize;
if (getCircular())
{
// Check if it is closer to scroll backwards
int32_t otherOffset = newOffset + getNumberOfItems() * itemSize;
int32_t offset = getOffset();
if (abs(otherOffset - offset) < abs(newOffset - offset))
{
newOffset = otherOffset;
}
}
return newOffset;
}
void ScrollWheelBase::animateToPosition(int32_t position, int16_t steps)
{
if (itemSize == 0)
{
return;
}
if (animateToCallback && animateToCallback->isValid() && itemSize > 0)
{
position = getNearestAlignedOffset(position);
int16_t itemIndex = (-position) / itemSize;
animateToCallback->execute(itemIndex);
}
ScrollBase::animateToPosition(position, steps);
}
int ScrollWheelBase::getSelectedItem() const
{
if (itemSize == 0)
{
return 0;
}
if (currentAnimationState == ANIMATING_GESTURE)
{
// Scroll in progress, get the destination value
return (-getNormalizedOffset(gestureEnd)) / itemSize;
}
return (-getNormalizedOffset(getOffset())) / itemSize;
}
int32_t ScrollWheelBase::keepOffsetInsideLimits(int32_t newOffset, int16_t overShoot) const
{
if (!getCircular())
{
newOffset = MIN(newOffset, overShoot);
int16_t numberOfItems = getNumberOfItems();
newOffset = MAX(newOffset, -(itemSize * (numberOfItems - 1)) - overShoot);
}
return newOffset;
}
void ScrollWheelBase::handleClickEvent(const ClickEvent& evt)
{
if (itemSize == 0)
{
return;
}
int32_t offset = getOffset();
if (evt.getType() == ClickEvent::PRESSED)
{
xClick = evt.getX();
yClick = evt.getY();
initialSwipeOffset = offset;
if (itemPressedCallback && itemPressedCallback->isValid())
{
itemPressedCallback->execute(getSelectedItem());
}
}
else if (evt.getType() == ClickEvent::RELEASED)
{
if (currentAnimationState == NO_ANIMATION)
{
int16_t click = getHorizontal() ? xClick : yClick;
// Click => move to clicked position
if (click < distanceBeforeAlignedItem)
{
animateToPosition(offset + ((distanceBeforeAlignedItem - click) / itemSize + 1) * itemSize);
}
else if (click > distanceBeforeAlignedItem + itemSize)
{
animateToPosition(offset - ((click - distanceBeforeAlignedItem) / itemSize) * itemSize);
}
else
{
animateToPosition(offset);
}
}
else if (currentAnimationState == ANIMATING_DRAG)
{
// click + drag + release. Find best Y to scroll to
animateToPosition(offset);
}
if (itemSelectedCallback && itemSelectedCallback->isValid())
{
itemSelectedCallback->execute(getSelectedItem());
}
}
}
void ScrollWheelBase::handleDragEvent(const DragEvent& evt)
{
currentAnimationState = ANIMATING_DRAG;
int newOffset = getOffset() + (getHorizontal() ? evt.getDeltaX() : evt.getDeltaY()) * dragAcceleration / 10;
if (!getCircular())
{
newOffset = MIN(newOffset, itemSize * 3 / 4);
int16_t numberOfItems = getNumberOfItems();
newOffset = MAX(newOffset, -(itemSize * (numberOfItems - 1)) - itemSize * 3 / 4);
}
setOffset(newOffset);
}
void ScrollWheelBase::handleGestureEvent(const GestureEvent& evt)
{
if (evt.getType() == (getHorizontal() ? GestureEvent::SWIPE_HORIZONTAL : GestureEvent::SWIPE_VERTICAL))
{
int32_t newOffset = getOffset() + evt.getVelocity() * swipeAcceleration / 10;
if (maxSwipeItems > 0)
{
int32_t maxDistance = maxSwipeItems * itemSize;
newOffset = MIN(newOffset, initialSwipeOffset + maxDistance);
newOffset = MAX(newOffset, initialSwipeOffset - maxDistance);
}
animateToPosition(newOffset);
}
}
void ScrollWheelBase::setAnimateToCallback(GenericCallback<int16_t>& callback)
{
animateToCallback = &callback;
}
} // namespace touchgfx

View File

@ -0,0 +1,203 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <touchgfx/containers/scrollers/ScrollWheelWithSelectionStyle.hpp>
namespace touchgfx
{
ScrollWheelWithSelectionStyle::ScrollWheelWithSelectionStyle()
: ScrollWheelBase(),
drawablesInFirstList(0),
extraSizeBeforeSelectedItem(0),
extraSizeAfterSelectedItem(0),
marginBeforeSelectedItem(0),
marginAfterSelectedItem(0),
drawables(0),
centerDrawables(0),
originalUpdateDrawableCallback(0),
originalUpdateCenterDrawableCallback(0)
{
ScrollWheelBase::add(list2);
ScrollWheelBase::add(list1); // Put center list at top of the first/last list.
}
void ScrollWheelWithSelectionStyle::setWidth(int16_t width)
{
ScrollWheelBase::setWidth(width);
if (getHorizontal())
{
refreshDrawableListsLayout();
}
}
void ScrollWheelWithSelectionStyle::setHeight(int16_t height)
{
ScrollWheelBase::setHeight(height);
if (!getHorizontal())
{
refreshDrawableListsLayout();
}
}
void ScrollWheelWithSelectionStyle::setHorizontal(bool horizontal)
{
ScrollWheelBase::setHorizontal(horizontal);
list1.setHorizontal(horizontal);
list2.setHorizontal(horizontal);
refreshDrawableListsLayout();
}
void ScrollWheelWithSelectionStyle::setCircular(bool circular)
{
ScrollWheelBase::setCircular(circular);
list1.setCircular(circular);
list2.setCircular(circular);
}
void ScrollWheelWithSelectionStyle::setNumberOfItems(int16_t numberOfItems)
{
if (numberOfItems != getNumberOfItems())
{
ScrollWheelBase::setNumberOfItems(numberOfItems);
list1.setNumberOfItems(numberOfItems);
list2.setNumberOfItems(numberOfItems);
}
}
void ScrollWheelWithSelectionStyle::setSelectedItemOffset(int16_t offset)
{
ScrollWheelBase::setSelectedItemOffset(offset);
refreshDrawableListsLayout();
}
void ScrollWheelWithSelectionStyle::setSelectedItemExtraSize(int16_t extraSizeBefore, int16_t extraSizeAfter)
{
extraSizeBeforeSelectedItem = extraSizeBefore;
extraSizeAfterSelectedItem = extraSizeAfter;
refreshDrawableListsLayout();
}
int16_t ScrollWheelWithSelectionStyle::getSelectedItemExtraSizeBefore() const
{
return extraSizeBeforeSelectedItem;
}
int16_t ScrollWheelWithSelectionStyle::getSelectedItemExtraSizeAfter() const
{
return extraSizeAfterSelectedItem;
}
void ScrollWheelWithSelectionStyle::setSelectedItemMargin(int16_t marginBefore, int16_t marginAfter)
{
marginBeforeSelectedItem = marginBefore;
marginAfterSelectedItem = marginAfter;
refreshDrawableListsLayout();
}
int16_t ScrollWheelWithSelectionStyle::getSelectedItemMarginBefore() const
{
return marginBeforeSelectedItem;
}
int16_t ScrollWheelWithSelectionStyle::getSelectedItemMarginAfter() const
{
return marginAfterSelectedItem;
}
void ScrollWheelWithSelectionStyle::setSelectedItemPosition(int16_t offset, int16_t extraSizeBefore, int16_t extraSizeAfter, int16_t marginBefore, int16_t marginAfter)
{
setSelectedItemOffset(offset);
setSelectedItemExtraSize(extraSizeBefore, extraSizeAfter);
setSelectedItemMargin(marginBefore, marginAfter);
}
void ScrollWheelWithSelectionStyle::setDrawableSize(int16_t drawableSize, int16_t drawableMargin)
{
ScrollWheelBase::setDrawableSize(drawableSize, drawableMargin);
list1.setDrawableSize(drawableSize, drawableMargin);
list2.setDrawableSize(drawableSize, drawableMargin);
// Resize the three lists
setSelectedItemOffset(distanceBeforeAlignedItem);
// Changing the drawable size alters number of required drawables, so refresh this
refreshDrawableListsLayout();
}
void ScrollWheelWithSelectionStyle::setDrawables(DrawableListItemsInterface& drawableListItems, GenericCallback<DrawableListItemsInterface*, int16_t, int16_t>& updateDrawableCallback,
DrawableListItemsInterface& centerDrawableListItems, GenericCallback<DrawableListItemsInterface*, int16_t, int16_t>& updateCenterDrawableCallback)
{
drawables = &drawableListItems;
centerDrawables = &centerDrawableListItems;
currentAnimationState = NO_ANIMATION;
originalUpdateDrawableCallback = &updateDrawableCallback;
originalUpdateCenterDrawableCallback = &updateCenterDrawableCallback;
refreshDrawableListsLayout();
setOffset(0);
}
void ScrollWheelWithSelectionStyle::setOffset(int32_t offset)
{
ScrollWheelBase::setOffset(offset);
list1.setOffset((distanceBeforeAlignedItem - (distanceBeforeAlignedItem - extraSizeBeforeSelectedItem)) + offset);
list2.setOffset((distanceBeforeAlignedItem - (distanceBeforeAlignedItem + itemSize + extraSizeAfterSelectedItem + marginAfterSelectedItem)) + offset);
}
void ScrollWheelWithSelectionStyle::itemChanged(int itemIndex)
{
ScrollWheelBase::itemChanged(itemIndex);
list1.itemChanged(itemIndex);
list2.itemChanged(itemIndex);
}
void ScrollWheelWithSelectionStyle::refreshDrawableListsLayout()
{
if (drawables != 0 && centerDrawables != 0)
{
int32_t currentOffset = getOffset();
int16_t list1Pos = distanceBeforeAlignedItem - extraSizeBeforeSelectedItem;
int16_t list2Pos = distanceBeforeAlignedItem + itemSize + (extraSizeAfterSelectedItem + marginAfterSelectedItem);
int16_t list0Size = list1Pos - marginBeforeSelectedItem;
int16_t list1Size = itemSize + extraSizeBeforeSelectedItem + extraSizeAfterSelectedItem;
if (getHorizontal())
{
int16_t list2Size = getWidth() - list2Pos;
list.setPosition(0, 0, list0Size, getHeight());
list1.setPosition(list1Pos, 0, list1Size, getHeight());
list2.setPosition(list2Pos, 0, list2Size, getHeight());
}
else
{
int16_t list2Size = getHeight() - list2Pos;
list.setPosition(0, 0, getWidth(), list0Size);
list1.setPosition(0, list1Pos, getWidth(), list1Size);
list2.setPosition(0, list2Pos, getWidth(), list2Size);
}
list.setDrawables(*drawables, 0, *originalUpdateDrawableCallback);
drawablesInFirstList = list.getNumberOfDrawables();
list1.setDrawables(*centerDrawables, 0, *originalUpdateCenterDrawableCallback);
list2.setDrawables(*drawables, drawablesInFirstList, *originalUpdateDrawableCallback);
setOffset(keepOffsetInsideLimits(currentOffset, 0));
}
}
} // namespace touchgfx

View File

@ -0,0 +1,35 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <touchgfx/widgets/AbstractButton.hpp>
namespace touchgfx
{
void AbstractButton::handleClickEvent(const ClickEvent& event)
{
bool wasPressed = pressed;
pressed = (event.getType() == ClickEvent::PRESSED);
if ((pressed && !wasPressed) || (!pressed && wasPressed))
{
// Pressed state changed, so invalidate
invalidate();
}
if (wasPressed && (event.getType() == ClickEvent::RELEASED))
{
// This is a click. Fire callback.
executeAction();
}
}
} // namespace touchgfx

View File

@ -0,0 +1,150 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <touchgfx/widgets/AnimatedImage.hpp>
namespace touchgfx
{
void AnimatedImage::handleTickEvent()
{
if (running)
{
++ticksSinceUpdate;
if (ticksSinceUpdate != updateTicksInterval)
{
return;
}
ticksSinceUpdate = 0;
BitmapId currentId = getBitmap();
if (((currentId == endId) && !reverse) || ((currentId == startId) && reverse))
{
if (!loopAnimation)
{
Application::getInstance()->unregisterTimerWidget(this);
running = false;
}
if (animationDoneAction && animationDoneAction->isValid())
{
animationDoneAction->execute(*this);
}
if (running && loopAnimation)
{
if (reverse)
{
Image::setBitmap(Bitmap(endId));
}
else
{
Image::setBitmap(Bitmap(startId));
}
invalidate();
}
}
else
{
if (reverse)
{
--currentId;
}
else
{
++currentId;
}
Image::setBitmap(Bitmap(currentId));
invalidate();
}
}
}
void AnimatedImage::startAnimation(const bool rev, const bool reset /*= false*/, const bool loop /*= false*/)
{
if ((startId != BITMAP_INVALID) && (endId != BITMAP_INVALID))
{
reverse = rev;
loopAnimation = loop;
if (reverse && reset)
{
Image::setBitmap(Bitmap(endId));
invalidate();
}
else if (!reverse && reset)
{
Image::setBitmap(Bitmap(startId));
invalidate();
}
Application::getInstance()->registerTimerWidget(this);
running = true;
}
}
void AnimatedImage::stopAnimation()
{
if (running)
{
Application::getInstance()->unregisterTimerWidget(this);
running = false;
}
if (reverse)
{
Image::setBitmap(Bitmap(endId));
}
else
{
Image::setBitmap(Bitmap(startId));
}
invalidate();
}
void AnimatedImage::pauseAnimation()
{
if (running)
{
Application::getInstance()->unregisterTimerWidget(this);
running = false;
}
else
{
Application::getInstance()->registerTimerWidget(this);
running = true;
}
}
void AnimatedImage::setBitmap(const Bitmap& bitmap)
{
startId = bitmap.getId();
Image::setBitmap(bitmap);
}
void AnimatedImage::setBitmapEnd(const Bitmap& bitmap)
{
endId = bitmap.getId();
}
void AnimatedImage::setBitmaps(BitmapId start, BitmapId end)
{
setBitmap(start);
setBitmapEnd(end);
}
void AnimatedImage::setUpdateTicksInterval(uint8_t updateInterval)
{
updateTicksInterval = updateInterval;
ticksSinceUpdate = 0;
}
} // namespace touchgfx

View File

@ -0,0 +1,190 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <touchgfx/widgets/AnimationTextureMapper.hpp>
namespace touchgfx
{
AnimationTextureMapper::AnimationTextureMapper()
: TextureMapper(),
textureMapperAnimationStepCallback(0),
textureMapperAnimationEndedCallback(0),
animationCounter(0),
animationRunning(false)
{
for (int i = 0; i < NUMBER_OF_ANIMATION_PARAMETERS; i++)
{
animations[i].animationActive = false;
}
}
void AnimationTextureMapper::setTextureMapperAnimationStepAction(GenericCallback<const AnimationTextureMapper&>& callback)
{
textureMapperAnimationStepCallback = &callback;
}
void AnimationTextureMapper::setTextureMapperAnimationEndedAction(GenericCallback<const AnimationTextureMapper&>& callback)
{
textureMapperAnimationEndedCallback = &callback;
}
bool AnimationTextureMapper::isTextureMapperAnimationRunning() const
{
return animationRunning;
}
void AnimationTextureMapper::setupAnimation(AnimationParameter parameter, float endValue, uint16_t duration, uint16_t delay, EasingEquation progressionEquation /*= &EasingEquations::linearEaseNone*/)
{
animations[parameter].animationActive = true;
animations[parameter].animationEnd = endValue;
animations[parameter].animationDuration = duration;
animations[parameter].animationDelay = delay;
animations[parameter].animationProgressionEquation = progressionEquation;
}
void AnimationTextureMapper::startAnimation()
{
Application::getInstance()->registerTimerWidget(this);
animationCounter = 0;
animations[X_ROTATION].animationStart = xAngle;
animations[Y_ROTATION].animationStart = yAngle;
animations[Z_ROTATION].animationStart = zAngle;
animations[SCALE].animationStart = scale;
animationRunning = true;
for (int i = 0; i < NUMBER_OF_ANIMATION_PARAMETERS; i++)
{
if (animations[i].animationActive && animations[i].animationDelay + animations[i].animationDuration > 0)
{
return; // Animation needs to run, return
}
}
// No active animations or all active animations have zero steps, execute now!
handleTickEvent();
}
void AnimationTextureMapper::cancelAnimationTextureMapperAnimation()
{
Application::getInstance()->unregisterTimerWidget(this);
animationRunning = false;
for (int i = 0; i < NUMBER_OF_ANIMATION_PARAMETERS; i++)
{
animations[i].animationActive = false;
}
}
uint16_t AnimationTextureMapper::getAnimationStep()
{
return animationCounter;
}
void AnimationTextureMapper::handleTickEvent()
{
if (animationRunning)
{
bool newValuesAssigned = false;
bool activeAnimationExists = false;
animationCounter++;
float newXAngle = xAngle;
float newYAngle = yAngle;
float newZAngle = zAngle;
float newScale = scale;
for (int i = 0; i < NUMBER_OF_ANIMATION_PARAMETERS; i++)
{
if (!(animations[i].animationActive))
{
continue;
}
if (animationCounter >= animations[i].animationDelay)
{
// Adjust the used animationCounter for the startup delay
uint32_t actualAnimationCounter = animationCounter - animations[i].animationDelay;
int directionModifier;
int16_t distance;
if (animations[i].animationEnd > animations[i].animationStart)
{
directionModifier = 1;
distance = (int16_t)((animations[i].animationEnd - animations[i].animationStart) * 1000);
}
else
{
directionModifier = -1;
distance = (int16_t)((animations[i].animationStart - animations[i].animationEnd) * 1000);
}
float delta = directionModifier * (animations[i].animationProgressionEquation(actualAnimationCounter, 0, distance, animations[i].animationDuration) / 1000.f);
switch ((AnimationParameter)i)
{
case X_ROTATION:
newXAngle = animations[X_ROTATION].animationStart + delta;
break;
case Y_ROTATION:
newYAngle = animations[Y_ROTATION].animationStart + delta;
break;
case Z_ROTATION:
newZAngle = animations[Z_ROTATION].animationStart + delta;
break;
case SCALE:
newScale = animations[SCALE].animationStart + delta;
break;
default:
break;
}
newValuesAssigned = true;
}
if (animationCounter >= (uint32_t)(animations[i].animationDelay + animations[i].animationDuration))
{
animations[i].animationActive = false;
}
else
{
activeAnimationExists = true;
}
}
if (newValuesAssigned)
{
updateAngles(newXAngle, newYAngle, newZAngle);
setScale(newScale);
if (textureMapperAnimationStepCallback && textureMapperAnimationStepCallback->isValid())
{
textureMapperAnimationStepCallback->execute(*this);
}
}
if (!activeAnimationExists)
{
// End of animation
cancelAnimationTextureMapperAnimation();
if (textureMapperAnimationEndedCallback && textureMapperAnimationEndedCallback->isValid())
{
textureMapperAnimationEndedCallback->execute(*this);
}
}
}
}
} // namespace touchgfx

View File

@ -0,0 +1,38 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <touchgfx/widgets/Box.hpp>
#include <touchgfx/hal/HAL.hpp>
namespace touchgfx
{
Rect Box::getSolidRect() const
{
Rect solidRect(0, 0, 0, 0);
if (alpha == 255)
{
solidRect.width = rect.width;
solidRect.height = rect.height;
}
return solidRect;
}
void Box::draw(const Rect& area) const
{
Rect dirty = area;
translateRectToAbsolute(dirty);
HAL::lcd().fillRect(dirty, color, alpha);
}
} // namespace touchgfx

View File

@ -0,0 +1,57 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <touchgfx/widgets/BoxWithBorder.hpp>
#include <touchgfx/hal/HAL.hpp>
namespace touchgfx
{
void BoxWithBorder::draw(const Rect& area) const
{
const Rect center = Rect(borderSize, borderSize, getWidth() - (2 * borderSize), getHeight() - (2 * borderSize));
if (center.isEmpty())
{
Rect dirty = area;
translateRectToAbsolute(dirty);
HAL::lcd().fillRect(dirty, borderColor, alpha);
return;
}
Rect dirty = area & center;
Box::draw(dirty);
if (borderSize == 0)
{
return;
}
Rect borders[4] =
{
Rect(0, 0, getWidth(), borderSize), // Upper
Rect(0, getHeight() - borderSize, getWidth(), borderSize), // lower
Rect(0, borderSize, borderSize, getHeight() - (2 * borderSize)), // left
Rect(getWidth() - borderSize, borderSize, borderSize, getHeight() - (2 * borderSize)) // right
};
for (int i = 0; i < 4; i++)
{
Rect borderDirty = borders[i] & area;
if (!borderDirty.isEmpty())
{
translateRectToAbsolute(borderDirty);
HAL::lcd().fillRect(borderDirty, borderColor, alpha);
}
}
}
} // namespace touchgfx

View File

@ -0,0 +1,52 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <touchgfx/widgets/Button.hpp>
#include <touchgfx/hal/HAL.hpp>
namespace touchgfx
{
void Button::draw(const Rect& invalidatedArea) const
{
Bitmap bmp(pressed ? down : up);
Rect dirty(0, 0, bmp.getWidth(), bmp.getHeight());
dirty &= invalidatedArea;
if ((bmp.getId() != BITMAP_INVALID) && !dirty.isEmpty())
{
Rect r;
translateRectToAbsolute(r);
HAL::lcd().drawPartialBitmap(bmp, r.x, r.y, dirty, alpha);
}
}
void Button::setBitmaps(const Bitmap& bitmapReleased, const Bitmap& bitmapPressed)
{
up = bitmapReleased;
down = bitmapPressed;
// Adjust width and height of this widget to match bitmap. It is assumed
// that the two bitmaps have same dimensions.
Button::setWidthHeight(down);
}
Rect Button::getSolidRect() const
{
if (alpha < 255)
{
return Rect(0, 0, 0, 0);
}
return (pressed ? down.getSolidRect() : up.getSolidRect());
}
} // namespace touchgfx

View File

@ -0,0 +1,56 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <touchgfx/widgets/ButtonWithIcon.hpp>
#include <touchgfx/hal/HAL.hpp>
namespace touchgfx
{
ButtonWithIcon::ButtonWithIcon()
: Button(),
iconX(0),
iconY(0)
{
}
void ButtonWithIcon::setBitmaps(const Bitmap& newBackgroundReleased, const Bitmap& newBackgroundPressed,
const Bitmap& newIconReleased, const Bitmap& newIconPressed)
{
Button::setBitmaps(newBackgroundReleased, newBackgroundPressed);
iconReleased = newIconReleased;
iconPressed = newIconPressed;
iconX = (getWidth() / 2) - (newIconPressed.getWidth() / 2);
iconY = (getHeight() / 2) - (newIconPressed.getHeight() / 2);
}
void ButtonWithIcon::draw(const Rect& invalidatedArea) const
{
Button::draw(invalidatedArea);
Bitmap bmp((pressed ? iconPressed : iconReleased));
Rect iconRect(iconX, iconY, bmp.getWidth(), bmp.getHeight());
Rect dirty = invalidatedArea & iconRect;
if ((bmp.getId() != BITMAP_INVALID) && !dirty.isEmpty())
{
Rect r;
translateRectToAbsolute(r);
dirty.x -= iconX;
dirty.y -= iconY;
HAL::lcd().drawPartialBitmap(bmp, r.x + iconX, r.y + iconY, dirty, alpha);
}
}
} // namespace touchgfx

View File

@ -0,0 +1,63 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <touchgfx/widgets/ButtonWithLabel.hpp>
#include <touchgfx/FontManager.hpp>
#include <touchgfx/hal/HAL.hpp>
namespace touchgfx
{
ButtonWithLabel::ButtonWithLabel()
: Button(), color(0), colorPressed(0), rotation(TEXT_ROTATE_0), textHeightIncludingSpacing(0)
{
}
void ButtonWithLabel::draw(const Rect& area) const
{
Button::draw(area);
if (typedText.hasValidId())
{
const Font* fontToDraw = typedText.getFont(); //never return 0
uint8_t height = textHeightIncludingSpacing;
int16_t offset;
Rect labelRect;
switch (rotation)
{
default:
case TEXT_ROTATE_0:
case TEXT_ROTATE_180:
offset = (this->getHeight() - height) / 2;
labelRect = Rect(0, offset, this->getWidth(), height);
break;
case TEXT_ROTATE_90:
case TEXT_ROTATE_270:
offset = (this->getWidth() - height) / 2;
labelRect = Rect(offset, 0, height, this->getHeight());
break;
}
Rect dirty = labelRect & area;
if (!dirty.isEmpty())
{
dirty.x -= labelRect.x;
dirty.y -= labelRect.y;
translateRectToAbsolute(labelRect);
LCD::StringVisuals visuals(fontToDraw, pressed ? colorPressed : color, alpha, typedText.getAlignment(), 0, rotation, typedText.getTextDirection(), 0, WIDE_TEXT_NONE);
HAL::lcd().drawString(labelRect, dirty, visuals, typedText.getText());
}
}
}
} // namespace touchgfx

View File

@ -0,0 +1,183 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <touchgfx/widgets/Gauge.hpp>
namespace touchgfx
{
Gauge::Gauge()
: AbstractProgressIndicator(),
algorithmMoving(TextureMapper::BILINEAR_INTERPOLATION),
algorithmSteady(TextureMapper::BILINEAR_INTERPOLATION),
needleStartAngle(0),
needleEndAngle(0),
gaugeCenterX(0),
gaugeCenterY(0),
needleCenterX(0),
needleCenterY(0)
{
remove(progressIndicatorContainer);
add(arc);
add(needle);
arc.setVisible(false);
}
void Gauge::setWidth(int16_t width)
{
AbstractProgressIndicator::setWidth(width);
needle.setWidth(width);
arc.setWidth(width);
}
void Gauge::setHeight(int16_t height)
{
AbstractProgressIndicator::setHeight(height);
needle.setHeight(height);
arc.setHeight(height);
}
void Gauge::setBackgroundOffset(int16_t offsetX, int16_t offsetY)
{
background.setXY(offsetX, offsetY);
}
void Gauge::setCenter(int x, int y)
{
gaugeCenterX = x;
gaugeCenterY = y;
setupNeedleTextureMapper();
arc.setPixelCenter(x - arc.getX(), y - arc.getY());
}
void Gauge::setArcPosition(int16_t x, int16_t y, int16_t width, int16_t height)
{
arc.setPosition(x, y, width, height);
arc.setPixelCenter(gaugeCenterX - x, gaugeCenterY - y);
}
void Gauge::setNeedle(const BitmapId bitmapId, int16_t rotationCenterX, int16_t rotationCenterY)
{
needle.setBitmap(Bitmap(bitmapId));
needleCenterX = rotationCenterX;
needleCenterY = rotationCenterY;
setupNeedleTextureMapper();
}
void Gauge::setMovingNeedleRenderingAlgorithm(TextureMapper::RenderingAlgorithm algorithm)
{
algorithmMoving = algorithm;
}
void Gauge::setSteadyNeedleRenderingAlgorithm(TextureMapper::RenderingAlgorithm algorithm)
{
algorithmSteady = algorithm;
}
void Gauge::setStartEndAngle(int startAngle, int endAngle)
{
assert(startAngle != endAngle);
needleStartAngle = startAngle;
needleEndAngle = endAngle;
arc.setArc(startAngle, endAngle);
Gauge::setValue(Gauge::getValue());
}
int Gauge::getStartAngle() const
{
return needleStartAngle;
}
int Gauge::getEndAngle() const
{
return needleEndAngle;
}
void Gauge::setArcVisible(bool show /*= true*/)
{
arc.setVisible(show);
}
void Gauge::putArcOnTop(bool onTop /*= true*/)
{
if (onTop)
{
remove(arc);
add(arc);
}
else
{
remove(needle);
add(needle);
}
}
touchgfx::Circle& Gauge::getArc()
{
return arc;
}
void Gauge::setValue(int value)
{
AbstractProgressIndicator::setValue(value);
if (animationStep >= animationDuration)
{
needle.setRenderingAlgorithm(algorithmSteady);
}
else
{
needle.setRenderingAlgorithm(algorithmMoving);
}
uint16_t progress = AbstractProgressIndicator::getProgress(abs(needleEndAngle - needleStartAngle));
if (needleEndAngle < needleStartAngle)
{
needle.updateZAngle((needleStartAngle - progress) / 180.0f * PI);
arc.updateArcEnd(needleStartAngle - progress);
}
else
{
needle.updateZAngle((needleStartAngle + progress) / 180.0f * PI);
arc.updateArcEnd(needleStartAngle + progress);
}
}
void Gauge::setAlpha(uint8_t newAlpha)
{
background.setAlpha(newAlpha);
needle.setAlpha(newAlpha);
arc.setAlpha(newAlpha);
}
uint8_t Gauge::getAlpha() const
{
return needle.getAlpha();
}
void Gauge::setupNeedleTextureMapper()
{
needle.setWidthHeight(*this);
needle.setXY(0, 0);
needle.setBitmapPosition(gaugeCenterX - needleCenterX, gaugeCenterY - needleCenterY);
needle.setCameraDistance(300.0f);
needle.setOrigo((float)gaugeCenterX, (float)gaugeCenterY, needle.getCameraDistance());
needle.setCamera(needle.getOrigoX(), needle.getOrigoY());
needle.setRenderingAlgorithm(TextureMapper::BILINEAR_INTERPOLATION);
}
void Gauge::setProgressIndicatorPosition(int16_t /*x*/, int16_t /*y*/, int16_t /*width*/, int16_t /*height*/)
{
}
} // namespace touchgfx

View File

@ -0,0 +1,52 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <touchgfx/widgets/Image.hpp>
#include <touchgfx/hal/HAL.hpp>
namespace touchgfx
{
void Image::setBitmap(const Bitmap& bitmap)
{
this->bitmap = bitmap;
// When setting bitmap, adjust size of this widget to match.
setWidthHeight(bitmap);
}
void Image::draw(const Rect& invalidatedArea) const
{
Rect meAbs;
translateRectToAbsolute(meAbs); //To find our x and y coords in absolute.
// Calculate intersection between bitmap rect and invalidated area.
Rect dirtyBitmapArea = bitmap.getRect() & invalidatedArea;
if (!dirtyBitmapArea.isEmpty())
{
HAL::lcd().drawPartialBitmap(bitmap, meAbs.x, meAbs.y, dirtyBitmapArea, alpha);
}
}
Rect Image::getSolidRect() const
{
// If alpha is less than solid, we have an empty solid rect.
if (alpha < 255)
{
return Rect(0, 0, 0, 0);
}
// Return solid rect from bitmap (precalculated).
return bitmap.getSolidRect();
}
} // namespace touchgfx

View File

@ -0,0 +1,296 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <touchgfx/hal/Types.hpp>
#include <touchgfx/widgets/Keyboard.hpp>
namespace touchgfx
{
Keyboard::Keyboard()
: Container(), keyListener(0), bufferSize(0), bufferPosition(0), highlightImage(), cancelIsEmitted(false)
{
setTouchable(true);
keyMappingList = static_cast<KeyMappingList*>(0);
buffer = static_cast<Unicode::UnicodeChar*>(0);
layout = static_cast<Layout*>(0);
image.setXY(0, 0);
Container::add(image);
highlightImage.setVisible(false);
Container::add(highlightImage);
enteredText.setColor(Color::getColorFrom24BitRGB(0, 0, 0));
Container::add(enteredText);
}
void Keyboard::setBuffer(Unicode::UnicodeChar* newBuffer, uint16_t newBufferSize)
{
buffer = newBuffer;
bufferSize = newBufferSize;
enteredText.setWildcard(buffer);
// Place cursor at end of string if we already have something
// in the edit buffer.
bufferPosition = Unicode::strlen(buffer);
}
void Keyboard::setLayout(const Layout* newLayout)
{
layout = newLayout;
if (newLayout != 0)
{
image.setBitmap(Bitmap(newLayout->bitmap));
enteredText.setTypedText(newLayout->textAreaFont);
enteredText.setColor(newLayout->textAreaFontColor);
enteredText.setPosition(newLayout->textAreaPosition.x, newLayout->textAreaPosition.y,
newLayout->textAreaPosition.width, newLayout->textAreaPosition.height);
}
invalidate();
}
void Keyboard::setTextIndentation()
{
if (layout != 0)
{
uint8_t indentation = layout->textAreaFont.getFont()->getMaxPixelsLeft();
enteredText.setPosition(layout->textAreaPosition.x - indentation, layout->textAreaPosition.y,
layout->textAreaPosition.width + indentation * 2, layout->textAreaPosition.height);
enteredText.setIndentation(indentation);
}
}
Keyboard::Key Keyboard::getKeyForCoordinates(int16_t x, int16_t y) const
{
Key key;
key.keyId = 0; // No key
if (layout != 0)
{
for (uint8_t i = 0; i < layout->numberOfKeys; i++)
{
if (layout->keyArray[i].keyArea.intersect(x, y))
{
key = layout->keyArray[i];
break;
}
}
}
return key;
}
Keyboard::CallbackArea Keyboard::getCallbackAreaForCoordinates(int16_t x, int16_t y) const
{
CallbackArea area;
area.callback = reinterpret_cast<GenericCallback<>*>(0);
if (layout != 0)
{
for (uint8_t i = 0; i < layout->numberOfCallbackAreas; i++)
{
if (layout->callbackAreaArray[i].keyArea.intersect(x, y))
{
area = layout->callbackAreaArray[i];
break;
}
}
}
return area;
}
void Keyboard::draw(const Rect& invalidatedArea) const
{
assert(layout && "No layout configured for Keyboard");
if (layout != 0)
{
Font* font = FontManager::getFont(layout->keyFont);
assert(font && "Keyboard::draw: Unable to find font, is font db initialized?");
if (font != 0)
{
// Setup visuals for h-center of "string"
LCD::StringVisuals visuals;
visuals.font = font;
visuals.alignment = CENTER;
visuals.color = layout->keyFontColor;
// String with room for a single character
Unicode::UnicodeChar character[2] = { 0, 0 }; // The last is important as string terminator.
uint16_t fontHeight = font->getMinimumTextHeight();
for (uint8_t i = 0; i < layout->numberOfKeys; i++)
{
const Key& key = layout->keyArray[i];
if (key.keyArea.intersect(invalidatedArea))
{
uint8_t keyId = key.keyId;
Unicode::UnicodeChar c = getCharForKey(keyId);
if (c != 0)
{
// Get a copy of the keyArea and v-center the area for the character
Rect keyArea = key.keyArea;
uint16_t offset = (keyArea.height - fontHeight) / 2;
keyArea.y += offset;
keyArea.height -= offset;
// Calculate the invalidated area relative to the key
Rect invalidatedAreaRelative = key.keyArea & invalidatedArea;
invalidatedAreaRelative.x -= keyArea.x;
invalidatedAreaRelative.y -= keyArea.y;
// Set up string with one character
character[0] = c;
translateRectToAbsolute(keyArea);
HAL::lcd().drawString(keyArea, invalidatedAreaRelative, visuals, character);
}
}
}
}
}
}
void Keyboard::handleClickEvent(const ClickEvent& evt)
{
ClickEvent::ClickEventType type = evt.getType();
if (type == ClickEvent::RELEASED && cancelIsEmitted)
{
cancelIsEmitted = false;
return;
}
int16_t x = evt.getX();
int16_t y = evt.getY();
Rect toDraw;
Keyboard::CallbackArea callbackArea = getCallbackAreaForCoordinates(x, y);
if (callbackArea.callback != 0)
{
if (type == ClickEvent::PRESSED)
{
highlightImage.setXY(callbackArea.keyArea.x, callbackArea.keyArea.y);
highlightImage.setBitmap(Bitmap(callbackArea.highlightBitmapId));
highlightImage.setVisible(true);
toDraw = highlightImage.getRect();
invalidateRect(toDraw);
}
if ((type == ClickEvent::RELEASED) && callbackArea.callback->isValid())
{
callbackArea.callback->execute();
if (keyListener)
{
keyListener->execute(0);
}
}
}
else
{
Keyboard::Key key = getKeyForCoordinates(x, y);
if (type == ClickEvent::PRESSED)
{
highlightImage.setXY(key.keyArea.x, key.keyArea.y);
highlightImage.setBitmap(Bitmap(key.highlightBitmapId));
highlightImage.setVisible(true);
toDraw = highlightImage.getRect();
invalidateRect(toDraw);
}
if (type == ClickEvent::RELEASED)
{
if (key.keyId != 0 && buffer)
{
Unicode::UnicodeChar c = getCharForKey(key.keyId);
if (c != 0)
{
uint16_t prevBufferPosition = bufferPosition;
if (bufferPosition < (bufferSize - 1))
{
buffer[bufferPosition++] = c;
buffer[bufferPosition] = 0;
}
if (prevBufferPosition != bufferPosition)
{
enteredText.invalidate();
if (keyListener)
{
keyListener->execute(c);
}
}
}
}
}
}
if (type == ClickEvent::RELEASED || type == ClickEvent::CANCEL)
{
toDraw = highlightImage.getRect();
highlightImage.setVisible(false);
invalidateRect(toDraw);
if (type == ClickEvent::CANCEL)
{
cancelIsEmitted = true;
}
}
}
void Keyboard::handleDragEvent(const DragEvent& event)
{
if (highlightImage.isVisible() && (!highlightImage.getRect().intersect(static_cast<int16_t>(event.getNewX()), static_cast<int16_t>(event.getNewY()))) && (!cancelIsEmitted))
{
// Send a CANCEL click event, if user has dragged out of currently pressed/highlighted key.
ClickEvent cancelEvent(ClickEvent::CANCEL, static_cast<int16_t>(event.getOldX()), static_cast<int16_t>(event.getOldY()));
handleClickEvent(cancelEvent);
}
}
Unicode::UnicodeChar Keyboard::getCharForKey(uint8_t keyId) const
{
Unicode::UnicodeChar ch = 0;
if (keyMappingList != 0)
{
for (uint8_t i = 0; i < keyMappingList->numberOfKeys; i++)
{
if (keyMappingList->keyMappingArray[i].keyId == keyId)
{
ch = keyMappingList->keyMappingArray[i].keyValue;
break;
}
}
}
return ch;
}
void Keyboard::setupDrawChain(const Rect& invalidatedArea, Drawable** nextPreviousElement)
{
// Keyboard is a Container, and they do not normally appear in the draw chain (they just draw children).
// But this particular container actually has a draw() function implementation, so we must change default
// behavior.
// First, add children
Container::setupDrawChain(invalidatedArea, nextPreviousElement);
// Then add yourself
Drawable::setupDrawChain(invalidatedArea, nextPreviousElement);
}
void Keyboard::setBufferPosition(uint16_t newPos)
{
bufferPosition = newPos;
enteredText.invalidate();
}
void Keyboard::setKeymappingList(const KeyMappingList* newKeyMappingList)
{
keyMappingList = newKeyMappingList;
invalidate();
}
} // namespace touchgfx

View File

@ -0,0 +1,92 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <touchgfx/hal/HAL.hpp>
#include <touchgfx/widgets/PixelDataWidget.hpp>
namespace touchgfx
{
PixelDataWidget::PixelDataWidget()
: Widget(),
buffer(0),
format(Bitmap::RGB888),
alpha(255)
{
}
void PixelDataWidget::setPixelData(uint8_t* const data)
{
buffer = data;
}
void PixelDataWidget::setBitmapFormat(Bitmap::BitmapFormat f)
{
format = f;
}
void PixelDataWidget::setAlpha(uint8_t newAlpha)
{
alpha = newAlpha;
}
uint8_t PixelDataWidget::getAlpha() const
{
return alpha;
}
void PixelDataWidget::draw(const Rect& invalidatedArea) const
{
if (buffer != 0)
{
//convert to lcd coordinates
const Rect absolute = getAbsoluteRect();
//copy to LCD
HAL::lcd().blitCopy(buffer, format, absolute, invalidatedArea, alpha, false);
}
}
Rect PixelDataWidget::getSolidRect() const
{
Rect solidRect(0, 0, 0, 0);
if (alpha == 255)
{
// There are at least some solid pixels
switch (format)
{
case Bitmap::BW: ///< 1-bit, black / white, no alpha channel
case Bitmap::BW_RLE: ///< 1-bit, black / white, no alpha channel compressed with horizontal RLE
case Bitmap::GRAY2: ///< 2-bit, gray scale, no alpha channel
case Bitmap::GRAY4: ///< 4-bit, gray scale, no alpha channel
case Bitmap::RGB565: ///< 16-bit, 5 bits for red, 6 bits for green, 5 bits for blue, no alpha channel
case Bitmap::RGB888: ///< 16-bit, 5 bits for red, 6 bits for green, 5 bits for blue, no alpha channel
//All solid pixels
solidRect.width = getWidth();
solidRect.height = getHeight();
break;
case Bitmap::ARGB8888: ///< 32-bit, 8 bits for each of red, green, blue and alpha channel
case Bitmap::ARGB2222: ///< 8-bit color
case Bitmap::ABGR2222: ///< 8-bit color
case Bitmap::RGBA2222: ///< 8-bit color
case Bitmap::BGRA2222: ///< 8-bit color
case Bitmap::L8: ///< 8-bit indexed color
case Bitmap::A4: ///< 4-bit alpha level
case Bitmap::CUSTOM: ///< Custom pixelformat
//No knowledge about solid pixels
break;
}
}
return solidRect;
}
} // namespace touchgfx

View File

@ -0,0 +1,97 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <touchgfx/widgets/RadioButton.hpp>
#include <touchgfx/hal/HAL.hpp>
namespace touchgfx
{
void RadioButton::draw(const Rect& invalidatedArea) const
{
Bitmap bitmap = getCurrentlyDisplayedBitmap();
if (bitmap.getId() != BITMAP_INVALID)
{
Rect meAbs;
translateRectToAbsolute(meAbs); //To find our x and y coords in absolute.
// Calculate intersection between bitmap rect and invalidated area.
Rect dirtyBitmapArea = bitmap.getRect() & invalidatedArea;
if (!dirtyBitmapArea.isEmpty())
{
HAL::lcd().drawPartialBitmap(bitmap, meAbs.x, meAbs.y, dirtyBitmapArea, alpha);
}
}
}
void RadioButton::handleClickEvent(const ClickEvent& event)
{
bool wasPressed = pressed;
pressed = (event.getType() == ClickEvent::PRESSED);
if ((pressed && !wasPressed) || (!pressed && wasPressed))
{
invalidate();
}
if (wasPressed && (event.getType() == ClickEvent::RELEASED))
{
if (deselectionEnabled)
{
setSelected(!getSelected());
}
else if (!getSelected())
{
setSelected(true);
}
}
}
void RadioButton::setBitmaps(const Bitmap& bmpUnselected, const Bitmap& bmpUnselectedPressed, const Bitmap& bmpSelected, const Bitmap& bmpSelectedPressed)
{
bitmapUnselected = bmpUnselected;
bitmapUnselectedPressed = bmpUnselectedPressed;
bitmapSelected = bmpSelected;
bitmapSelectedPressed = bmpSelectedPressed;
RadioButton::setWidthHeight(bitmapUnselected);
}
Rect RadioButton::getSolidRect() const
{
if (alpha < 255)
{
return Rect(0, 0, 0, 0);
}
return getCurrentlyDisplayedBitmap().getSolidRect();
}
void RadioButton::setSelected(bool newSelected)
{
bool wasSelected = selected;
selected = newSelected;
if (wasSelected && !newSelected)
{
executeDeselectedAction();
}
if (!wasSelected && newSelected)
{
executeAction();
}
invalidate();
}
} // namespace touchgfx

View File

@ -0,0 +1,83 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <touchgfx/widgets/RepeatButton.hpp>
namespace touchgfx
{
RepeatButton::RepeatButton()
: Button(), ticksDelay(30), ticksInterval(15), ticks(0), ticksBeforeContinuous(0)
{
}
void RepeatButton::setDelay(int delay)
{
ticksDelay = delay;
}
int RepeatButton::getDelay()
{
return ticksDelay;
}
void RepeatButton::setInterval(int interval)
{
ticksInterval = interval;
}
int RepeatButton::getInterval()
{
return ticksInterval;
}
void RepeatButton::handleClickEvent(const ClickEvent& event)
{
pressed = false; // To prevent AbstractButton from calling action->execute().
invalidate(); // Force redraw after forced state change
Button::handleClickEvent(event);
if (event.getType() == ClickEvent::PRESSED)
{
executeAction();
ticks = 0;
ticksBeforeContinuous = ticksDelay;
Application::getInstance()->registerTimerWidget(this);
}
else
{
Application::getInstance()->unregisterTimerWidget(this);
}
}
void RepeatButton::handleTickEvent()
{
Button::handleTickEvent();
if (pressed)
{
if (ticks == ticksBeforeContinuous)
{
executeAction();
ticks = 0;
ticksBeforeContinuous = ticksInterval;
}
else
{
ticks++;
}
}
}
} // namespace touchgfx

View File

@ -0,0 +1,194 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <touchgfx/TextureMapTypes.hpp>
#include <touchgfx/hal/HAL.hpp>
#include <touchgfx/transforms/DisplayTransformation.hpp>
#include <touchgfx/widgets/ScalableImage.hpp>
namespace touchgfx
{
ScalableImage::ScalableImage(const Bitmap& bitmap /*= Bitmap() */)
: Image(bitmap),
currentScalingAlgorithm(BILINEAR_INTERPOLATION)
{
}
void ScalableImage::setScalingAlgorithm(ScalingAlgorithm algorithm)
{
currentScalingAlgorithm = algorithm;
}
ScalableImage::ScalingAlgorithm ScalableImage::getScalingAlgorithm()
{
return currentScalingAlgorithm;
}
void ScalableImage::drawQuad(const Rect& invalidatedArea, uint16_t* fb, const float* triangleXs, const float* triangleYs, const float* triangleZs, const float* triangleUs, const float* triangleVs) const
{
// Area to redraw. Relative to the scalableImage.
Rect dirtyArea = Rect(0, 0, getWidth(), getHeight()) & invalidatedArea;
// Absolute position of the scalableImage.
Rect dirtyAreaAbsolute = dirtyArea;
translateRectToAbsolute(dirtyAreaAbsolute);
Rect absoluteRect = getAbsoluteRect();
DisplayTransformation::transformDisplayToFrameBuffer(absoluteRect);
// Transform rects to match framebuffer coordinates
// This is needed if the display is rotated compared to the framebuffer
DisplayTransformation::transformDisplayToFrameBuffer(dirtyArea, this->getRect());
DisplayTransformation::transformDisplayToFrameBuffer(dirtyAreaAbsolute);
// Get a pointer to the bitmap data, return if no bitmap found
const uint16_t* textmap = (const uint16_t*)bitmap.getData();
if (!textmap)
{
return;
}
float x0 = triangleXs[0];
float x1 = triangleXs[1];
float x2 = triangleXs[2];
float x3 = triangleXs[3];
float y0 = triangleYs[0]; //lint !e578
float y1 = triangleYs[1]; //lint !e578
float y2 = triangleYs[2];
float y3 = triangleYs[3];
DisplayTransformation::transformDisplayToFrameBuffer(x0, y0, this->getRect());
DisplayTransformation::transformDisplayToFrameBuffer(x1, y1, this->getRect());
DisplayTransformation::transformDisplayToFrameBuffer(x2, y2, this->getRect());
DisplayTransformation::transformDisplayToFrameBuffer(x3, y3, this->getRect());
Point3D vertices[4];
Point3D point0 = { floatToFixed28_4(x0), floatToFixed28_4(y0), (float)(triangleZs[0]), (float)(triangleUs[0]), (float)(triangleVs[0]) };
Point3D point1 = { floatToFixed28_4(x1), floatToFixed28_4(y1), (float)(triangleZs[1]), (float)(triangleUs[1]), (float)(triangleVs[1]) };
Point3D point2 = { floatToFixed28_4(x2), floatToFixed28_4(y2), (float)(triangleZs[2]), (float)(triangleUs[2]), (float)(triangleVs[2]) };
Point3D point3 = { floatToFixed28_4(x3), floatToFixed28_4(y3), (float)(triangleZs[3]), (float)(triangleUs[3]), (float)(triangleVs[3]) };
vertices[0] = point0;
vertices[1] = point1;
vertices[2] = point2;
vertices[3] = point3;
DrawingSurface dest = { fb, HAL::FRAME_BUFFER_WIDTH };
TextureSurface src = { textmap, bitmap.getExtraData(), bitmap.getWidth(), bitmap.getHeight(), bitmap.getWidth() };
HAL::lcd().drawTextureMapQuad(dest, vertices, src, absoluteRect, dirtyAreaAbsolute, lookupRenderVariant(), alpha, 0xFFFF);
}
RenderingVariant ScalableImage::lookupRenderVariant() const
{
RenderingVariant renderVariant;
if (currentScalingAlgorithm == NEAREST_NEIGHBOR)
{
renderVariant = lookupNearestNeighborRenderVariant(bitmap);
}
else
{
renderVariant = lookupBilinearRenderVariant(bitmap);
}
return renderVariant;
}
void ScalableImage::draw(const Rect& invalidatedArea) const
{
if (!alpha)
{
return;
}
uint16_t* fb = 0;
float triangleXs[4];
float triangleYs[4];
float triangleZs[4];
float triangleUs[4];
float triangleVs[4];
float imageX0 = 0;
float imageY0 = 0;
float imageX1 = imageX0 + getWidth();
float imageY1 = imageY0;
float imageX2 = imageX1;
float imageY2 = imageY0 + getHeight();
float imageX3 = imageX0;
float imageY3 = imageY2;
triangleZs[0] = 100.f;
triangleZs[1] = 100.f;
triangleZs[2] = 100.f;
triangleZs[3] = 100.f;
// Setup texture coordinates
float right = (float)(bitmap.getWidth());
float bottom = (float)(bitmap.getHeight());
float textureU0 = 0.0f;
float textureV0 = 0.0f;
float textureU1 = right;
float textureV1 = 0.0f;
float textureU2 = right;
float textureV2 = bottom;
float textureU3 = 0.0f;
float textureV3 = bottom;
if (HAL::DISPLAY_ROTATION == rotate90)
{
textureU0 = 0.0f;
textureV0 = right;
textureU1 = 0.0f;
textureV1 = 0.0f;
textureU2 = bottom;
textureV2 = 0.0f;
textureU3 = bottom;
textureV3 = right;
}
triangleXs[0] = imageX0;
triangleXs[1] = imageX1;
triangleXs[2] = imageX2;
triangleXs[3] = imageX3;
triangleYs[0] = imageY0;
triangleYs[1] = imageY1;
triangleYs[2] = imageY2;
triangleYs[3] = imageY3;
triangleUs[0] = textureU0;
triangleUs[1] = textureU1;
triangleUs[2] = textureU2;
triangleUs[3] = textureU3;
triangleVs[0] = textureV0;
triangleVs[1] = textureV1;
triangleVs[2] = textureV2;
triangleVs[3] = textureV3;
drawQuad(invalidatedArea, fb, triangleXs, triangleYs, triangleZs, triangleUs, triangleVs);
}
Rect ScalableImage::getSolidRect() const
{
if (alpha < 255)
{
return Rect(0, 0, 0, 0);
}
// If original image is completely solid the scaled image will also be
if ((bitmap.getSolidRect().width == bitmap.getWidth()) && (bitmap.getSolidRect().height == bitmap.getHeight()))
{
return bitmap.getSolidRect();
}
return Rect(0, 0, 0, 0);
}
} // namespace touchgfx

View File

@ -0,0 +1,62 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <touchgfx/widgets/SnapshotWidget.hpp>
#include <touchgfx/hal/HAL.hpp>
namespace touchgfx
{
SnapshotWidget::SnapshotWidget()
: Widget(), bitmapId(BITMAP_INVALID), alpha(255)
{
}
void SnapshotWidget::draw(const Rect& invalidatedArea) const
{
if (alpha == 0 || bitmapId == BITMAP_INVALID)
{
return;
}
Rect absRect(0, 0, Bitmap(bitmapId).getWidth(), rect.height);
translateRectToAbsolute(absRect);
HAL::lcd().blitCopy((const uint16_t*)Bitmap(bitmapId).getData(), absRect, invalidatedArea, alpha, false);
}
Rect SnapshotWidget::getSolidRect() const
{
if (alpha < 255 || bitmapId == BITMAP_INVALID)
{
return Rect(0, 0, 0, 0);
}
else
{
return Rect(0, 0, getWidth(), getHeight());
}
}
void SnapshotWidget::makeSnapshot()
{
makeSnapshot(BITMAP_ANIMATION_STORAGE);
}
void SnapshotWidget::makeSnapshot(const BitmapId bmp)
{
Rect visRect(0, 0, rect.width, rect.height);
getVisibleRect(visRect);
Rect absRect = getAbsoluteRect();
bitmapId = (HAL::lcd().copyFrameBufferRegionToMemory(visRect, absRect, bmp)) ? bmp : BITMAP_INVALID;
}
} // namespace touchgfx

View File

@ -0,0 +1,193 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <touchgfx/widgets/TextArea.hpp>
#include <touchgfx/hal/HAL.hpp>
namespace touchgfx
{
int16_t TextArea::getTextHeight()
{
return typedText.hasValidId() ? calculateTextHeight(typedText.getText(), 0, 0) : 0;
}
uint16_t TextArea::getTextWidth() const
{
return typedText.hasValidId() ? typedText.getFont()->getStringWidth(typedText.getTextDirection(), typedText.getText(), 0, 0) : 0;
}
void TextArea::draw(const Rect& area) const
{
if (typedText.hasValidId())
{
const Font* fontToDraw = typedText.getFont();
if (fontToDraw != 0)
{
LCD::StringVisuals visuals(fontToDraw, color, alpha, typedText.getAlignment(), linespace, rotation, typedText.getTextDirection(), indentation, wideTextAction);
HAL::lcd().drawString(getAbsoluteRect(), area, visuals, typedText.getText(), 0, 0);
}
}
}
void TextArea::setTypedText(TypedText t)
{
typedText = t;
// If this TextArea does not yet have a width and height,
// just assign the smallest possible size to fit current text.
if (getWidth() == 0 && getHeight() == 0)
{
resizeToCurrentText();
}
}
void TextArea::resizeToCurrentText()
{
if (typedText.hasValidId())
{
uint16_t w = getTextWidth();
uint16_t h = getTextHeight();
if (rotation == TEXT_ROTATE_0 || rotation == TEXT_ROTATE_180)
{
setWidthHeight(w, h);
}
else
{
setWidthHeight(h, w);
}
}
}
void TextArea::resizeToCurrentTextWithAlignment()
{
if (typedText.hasValidId())
{
Alignment alignment = typedText.getAlignment();
uint16_t text_width = getTextWidth();
uint16_t text_height = getTextHeight();
if (rotation == TEXT_ROTATE_0 || rotation == TEXT_ROTATE_180)
{
// (rotate-0 && left-align) or (rotate-180 && right-align) places text to the left
if (!((rotation == TEXT_ROTATE_0 && alignment == LEFT) || (rotation == TEXT_ROTATE_180 && alignment == RIGHT)))
{
uint16_t old_width = getWidth();
uint16_t old_x = getX();
if (alignment == CENTER)
{
setX(old_x + (old_width - text_width) / 2);
}
else
{
setX(old_x + (old_width - text_width));
}
}
if (rotation == TEXT_ROTATE_180)
{
uint16_t old_height = getHeight();
uint16_t old_y = getY();
setY(old_y + (old_height - text_height));
}
setWidthHeight(text_width, text_height);
}
else
{
// 90+left or 270+right places text at the same y coordinate
if (!((rotation == TEXT_ROTATE_90 && alignment == LEFT) || (rotation == TEXT_ROTATE_270 && alignment == RIGHT)))
{
uint16_t old_height = getHeight();
uint16_t old_y = getY();
if (alignment == CENTER)
{
setY(old_y + (old_height - text_width) / 2);
}
else
{
setY(old_y + (old_height - text_width));
}
}
if (rotation == TEXT_ROTATE_90)
{
uint16_t old_width = getWidth();
uint16_t old_x = getX();
setX(old_x + (old_width - text_height));
}
setWidthHeight(text_height, text_width);
}
}
}
void TextArea::resizeHeightToCurrentText()
{
if (typedText.hasValidId())
{
uint16_t h = getTextHeight();
if (rotation == TEXT_ROTATE_0 || rotation == TEXT_ROTATE_180)
{
setHeight(h);
}
else
{
setWidth(h);
}
}
}
void TextArea::resizeHeightToCurrentTextWithRotation()
{
if (typedText.hasValidId())
{
uint16_t h = getTextHeight();
switch (rotation)
{
default:
case TEXT_ROTATE_0:
setHeight(h);
break;
case TEXT_ROTATE_90:
setX(rect.right() - h);
setWidth(h);
break;
case TEXT_ROTATE_180:
setY(rect.bottom() - h);
setHeight(h);
break;
case TEXT_ROTATE_270:
setWidth(h);
break;
}
}
}
int16_t TextArea::calculateTextHeight(const Unicode::UnicodeChar* format, ...) const
{
if (!typedText.hasValidId())
{
return 0;
}
va_list pArg;
va_start(pArg, format);
const Font* fontToDraw = typedText.getFont();
int16_t textHeight = fontToDraw->getMinimumTextHeight();
TextProvider textProvider;
textProvider.initialize(format, pArg, fontToDraw->getGSUBTable());
int16_t numLines = LCD::getNumLines(textProvider, wideTextAction, typedText.getTextDirection(), typedText.getFont(), getWidth());
va_end(pArg);
return (textHeight + linespace > 0) ? (numLines * textHeight + (numLines - 1) * linespace) : (numLines > 0) ? (textHeight) : 0;
}
} // namespace touchgfx

View File

@ -0,0 +1,46 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <touchgfx/widgets/TextAreaWithWildcard.hpp>
namespace touchgfx
{
void TextAreaWithOneWildcard::draw(const Rect& area) const
{
if (typedText.hasValidId())
{
const Font* fontToDraw = typedText.getFont();
if (fontToDraw != 0)
{
LCD::StringVisuals visuals(fontToDraw, color, alpha, typedText.getAlignment(), linespace, rotation, typedText.getTextDirection(), indentation, wideTextAction);
HAL::lcd().drawString(getAbsoluteRect(), area, visuals, typedText.getText(), wildcard, 0);
}
}
}
void TextAreaWithTwoWildcards::draw(const Rect& area) const
{
if (typedText.hasValidId())
{
const Font* fontToDraw = typedText.getFont();
if (fontToDraw != 0)
{
LCD::StringVisuals visuals(fontToDraw, color, alpha, typedText.getAlignment(), linespace, rotation, typedText.getTextDirection(), indentation, wideTextAction);
HAL::lcd().drawString(getAbsoluteRect(), area, visuals, typedText.getText(), wc1, wc2);
}
}
}
} // namespace touchgfx

View File

@ -0,0 +1,350 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <touchgfx/Math3D.hpp>
#include <touchgfx/TextureMapTypes.hpp>
#include <touchgfx/hal/HAL.hpp>
#include <touchgfx/transforms/DisplayTransformation.hpp>
#include <touchgfx/widgets/TextureMapper.hpp>
namespace touchgfx
{
TextureMapper::TextureMapper(const Bitmap& bitmap /*= Bitmap()*/)
: Image(bitmap),
currentRenderingAlgorithm(NEAREST_NEIGHBOR),
xBitmapPosition(0.0f),
yBitmapPosition(0.0f),
xAngle(0.0f),
yAngle(0.0f),
zAngle(0.0f),
scale(1.0f),
xOrigo(0.0f),
yOrigo(0.0f),
zOrigo(1000.0f),
xCamera(0.0f),
yCamera(0.0f),
cameraDistance(1000.0f),
imageX0(0.0f),
imageY0(0.0f),
imageZ0(1.0f),
imageX1(0.0f),
imageY1(0.0f),
imageZ1(1.0f),
imageX2(0.0f),
imageY2(0.0f),
imageZ2(1.0f),
imageX3(0.0f),
imageY3(0.0f),
imageZ3(1.0f),
subDivisionSize(12)
{
}
void TextureMapper::setBitmap(const Bitmap& bitmap)
{
Image::setBitmap(bitmap);
applyTransformation();
}
void TextureMapper::applyTransformation()
{
const uint8_t n = 4;
int imgWidth = Bitmap(bitmap).getWidth() + 1;
int imgHeight = Bitmap(bitmap).getHeight() + 1;
Point4 vertices[n] =
{
Point4(xBitmapPosition - 1, yBitmapPosition - 1, cameraDistance),
Point4(xBitmapPosition - 1 + imgWidth, yBitmapPosition - 1, cameraDistance),
Point4(xBitmapPosition - 1 + imgWidth, yBitmapPosition - 1 + imgHeight, cameraDistance),
Point4(xBitmapPosition - 1, yBitmapPosition - 1 + imgHeight, cameraDistance),
};
Point4 transformed[n];
Vector4 center(xOrigo, yOrigo, zOrigo);
Matrix4x4 translateToCenter;
translateToCenter.concatenateXTranslation(-center.getX()).concatenateYTranslation(-center.getY()).concatenateZTranslation(-center.getZ());
Matrix4x4 rotateAroundCenter;
rotateAroundCenter.concatenateXRotation(xAngle).concatenateYRotation(yAngle).concatenateZRotation(zAngle);
Matrix4x4 scaleAroundCenter;
scaleAroundCenter.concatenateXScale(scale).concatenateYScale(scale).concatenateZScale(scale);
Matrix4x4 translateFromCenter;
translateFromCenter.concatenateXTranslation(center.getX()).concatenateYTranslation(center.getY()).concatenateZTranslation(center.getZ());
Matrix4x4 transform = translateFromCenter * scaleAroundCenter * rotateAroundCenter * translateToCenter;
Matrix4x4 translateToCamera;
translateToCamera.concatenateXTranslation(-xCamera);
translateToCamera.concatenateYTranslation(-yCamera);
Matrix4x4 perspectiveProject;
perspectiveProject.setViewDistance(cameraDistance);
Matrix4x4 translateFromCamera;
translateFromCamera.concatenateXTranslation(xCamera).concatenateYTranslation(yCamera);
transform = translateFromCamera * perspectiveProject * translateToCamera * transform;
for (int i = 0; i < n; i++)
{
transformed[i] = transform * vertices[i];
}
imageX0 = ((float)transformed[0].getX() * cameraDistance / (float)transformed[0].getZ());
imageY0 = ((float)transformed[0].getY() * cameraDistance / (float)transformed[0].getZ());
imageZ0 = ((float)transformed[0].getZ());
imageX1 = ((float)transformed[1].getX() * cameraDistance / (float)transformed[1].getZ());
imageY1 = ((float)transformed[1].getY() * cameraDistance / (float)transformed[1].getZ());
imageZ1 = ((float)transformed[1].getZ());
imageX2 = ((float)transformed[2].getX() * cameraDistance / (float)transformed[2].getZ());
imageY2 = ((float)transformed[2].getY() * cameraDistance / (float)transformed[2].getZ());
imageZ2 = ((float)transformed[2].getZ());
imageX3 = ((float)transformed[3].getX() * cameraDistance / (float)transformed[3].getZ());
imageY3 = ((float)transformed[3].getY() * cameraDistance / (float)transformed[3].getZ());
imageZ3 = ((float)transformed[3].getZ());
}
Rect TextureMapper::getBoundingRect() const
{
// MIN and MAX are macros so do not insert them into each other
float minXf = MIN(imageX0, imageX1);
minXf = MIN(minXf, imageX2);
minXf = floorf(MIN(minXf, imageX3));
int16_t minX = (int16_t)(MAX(0, minXf));
float maxXf = MAX(imageX0, imageX1);
maxXf = MAX(maxXf, imageX2);
maxXf = ceilf(MAX(maxXf, imageX3));
int16_t maxX = getWidth();
maxX = (int16_t)(MIN(maxX, maxXf));
float minYf = MIN(imageY0, imageY1);
minYf = MIN(minYf, imageY2);
minYf = floorf(MIN(minYf, imageY3));
int16_t minY = (int16_t)(MAX(0, minYf));
float maxYf = MAX(imageY0, imageY1);
maxYf = MAX(maxYf, imageY2);
maxYf = ceilf(MAX(maxYf, imageY3));
int16_t maxY = getHeight();
maxY = (int16_t)(MIN(maxY, maxYf));
return Rect(minX, minY, maxX - minX, maxY - minY);
}
void TextureMapper::updateAngles(float newXAngle, float newYAngle, float newZAngle)
{
Rect rBefore = getBoundingRect();
xAngle = newXAngle;
yAngle = newYAngle;
zAngle = newZAngle;
applyTransformation();
Rect rAfter = getBoundingRect();
rAfter.expandToFit(rBefore);
invalidateRect(rAfter);
}
void TextureMapper::setScale(float _scale)
{
this->scale = _scale;
applyTransformation();
}
void TextureMapper::draw(const Rect& invalidatedArea) const
{
if (!alpha)
{
return;
}
uint16_t* fb = 0;
// Setup texture coordinates
float right = (float)(bitmap.getWidth());
float bottom = (float)(bitmap.getHeight());
float textureU0 = -1.0f;
float textureV0 = -1.0f;
float textureU1 = right;
float textureV1 = -1.0f;
float textureU2 = right;
float textureV2 = bottom;
float textureU3 = -1.0f;
float textureV3 = bottom;
if (HAL::DISPLAY_ROTATION == rotate90)
{
textureU0 = -1.0f;
textureV0 = right;
textureU1 = -1.0f;
textureV1 = -1.0f;
textureU2 = bottom;
textureV2 = -1.0f;
textureU3 = bottom;
textureV3 = right;
}
float triangleXs[4];
float triangleYs[4];
float triangleZs[4];
float triangleUs[4];
float triangleVs[4];
// Determine winding order
Vector4 zeroToOne(imageX1 - imageX0, imageY1 - imageY0, imageZ1 - imageZ0);
Vector4 zeroToTwo(imageX2 - imageX0, imageY2 - imageY0, imageZ2 - imageZ0);
Vector4 normal = zeroToOne.crossProduct(zeroToTwo);
if (normal.getZ() > 0)
{
triangleXs[0] = imageX0;
triangleXs[1] = imageX1;
triangleXs[2] = imageX2;
triangleXs[3] = imageX3;
triangleYs[0] = imageY0;
triangleYs[1] = imageY1;
triangleYs[2] = imageY2;
triangleYs[3] = imageY3;
triangleZs[0] = imageZ0;
triangleZs[1] = imageZ1;
triangleZs[2] = imageZ2;
triangleZs[3] = imageZ3;
triangleUs[0] = textureU0;
triangleUs[1] = textureU1;
triangleUs[2] = textureU2;
triangleUs[3] = textureU3;
triangleVs[0] = textureV0;
triangleVs[1] = textureV1;
triangleVs[2] = textureV2;
triangleVs[3] = textureV3;
}
else
{
// invert due to the triangles winding order (showing backface of the triangle)
triangleXs[1] = imageX0;
triangleXs[0] = imageX1;
triangleXs[2] = imageX3;
triangleXs[3] = imageX2;
triangleYs[1] = imageY0;
triangleYs[0] = imageY1;
triangleYs[2] = imageY3;
triangleYs[3] = imageY2;
triangleZs[1] = imageZ0;
triangleZs[0] = imageZ1;
triangleZs[2] = imageZ3;
triangleZs[3] = imageZ2;
triangleUs[1] = textureU0;
triangleUs[0] = textureU1;
triangleUs[2] = textureU3;
triangleUs[3] = textureU2;
triangleVs[1] = textureV0;
triangleVs[0] = textureV1;
triangleVs[2] = textureV3;
triangleVs[3] = textureV2;
}
drawQuad(invalidatedArea, fb, triangleXs, triangleYs, triangleZs, triangleUs, triangleVs);
}
void TextureMapper::drawQuad(const Rect& invalidatedArea, uint16_t* fb, const float* triangleXs, const float* triangleYs, const float* triangleZs, const float* triangleUs, const float* triangleVs) const
{
// Area to redraw. Relative to the TextureMapper.
Rect dirtyArea = Rect(0, 0, getWidth(), getHeight()) & invalidatedArea;
// Absolute position of the TextureMapper.
Rect dirtyAreaAbsolute = dirtyArea;
translateRectToAbsolute(dirtyAreaAbsolute);
Rect absoluteRect = getAbsoluteRect();
DisplayTransformation::transformDisplayToFrameBuffer(absoluteRect);
// Transform rects to match framebuffer coordinates
// This is needed if the display is rotated compared to the framebuffer
DisplayTransformation::transformDisplayToFrameBuffer(dirtyArea, this->getRect());
DisplayTransformation::transformDisplayToFrameBuffer(dirtyAreaAbsolute);
// Get a pointer to the bitmap data, return if no bitmap found
const uint16_t* textmap = (const uint16_t*)bitmap.getData();
if (!textmap)
{
return;
}
float x0 = triangleXs[0];
float x1 = triangleXs[1];
float x2 = triangleXs[2];
float x3 = triangleXs[3];
float y0 = triangleYs[0]; //lint !e578
float y1 = triangleYs[1]; //lint !e578
float y2 = triangleYs[2];
float y3 = triangleYs[3];
DisplayTransformation::transformDisplayToFrameBuffer(x0, y0, this->getRect());
DisplayTransformation::transformDisplayToFrameBuffer(x1, y1, this->getRect());
DisplayTransformation::transformDisplayToFrameBuffer(x2, y2, this->getRect());
DisplayTransformation::transformDisplayToFrameBuffer(x3, y3, this->getRect());
Point3D vertices[4];
Point3D point0 = { floatToFixed28_4(x0), floatToFixed28_4(y0), (float)(triangleZs[0]), (float)(triangleUs[0]), (float)(triangleVs[0]) };
Point3D point1 = { floatToFixed28_4(x1), floatToFixed28_4(y1), (float)(triangleZs[1]), (float)(triangleUs[1]), (float)(triangleVs[1]) };
Point3D point2 = { floatToFixed28_4(x2), floatToFixed28_4(y2), (float)(triangleZs[2]), (float)(triangleUs[2]), (float)(triangleVs[2]) };
Point3D point3 = { floatToFixed28_4(x3), floatToFixed28_4(y3), (float)(triangleZs[3]), (float)(triangleUs[3]), (float)(triangleVs[3]) };
vertices[0] = point0;
vertices[1] = point1;
vertices[2] = point2;
vertices[3] = point3;
DrawingSurface dest = { fb, HAL::FRAME_BUFFER_WIDTH };
TextureSurface src = { textmap, bitmap.getExtraData(), bitmap.getWidth(), bitmap.getHeight(), bitmap.getWidth() };
uint16_t subDivs = subDivisionSize;
if (point0.Z == point1.Z && point1.Z == point2.Z)
{
subDivs = 0xFFFF; // Max: One sweep
}
HAL::lcd().drawTextureMapQuad(dest, vertices, src, absoluteRect, dirtyAreaAbsolute, lookupRenderVariant(), alpha, subDivs);
}
RenderingVariant TextureMapper::lookupRenderVariant() const
{
RenderingVariant renderVariant;
if (currentRenderingAlgorithm == NEAREST_NEIGHBOR)
{
renderVariant = lookupNearestNeighborRenderVariant(bitmap);
}
else
{
renderVariant = lookupBilinearRenderVariant(bitmap);
}
return renderVariant;
}
Rect TextureMapper::getSolidRect() const
{
return Rect(0, 0, 0, 0);
}
} // namespace touchgfx

View File

@ -0,0 +1,155 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <touchgfx/widgets/TiledImage.hpp>
#include <touchgfx/hal/HAL.hpp>
namespace touchgfx
{
void TiledImage::setBitmap(const Bitmap& bitmap)
{
Image::setBitmap(bitmap);
// Make sure the xOffset and yOffset are correct (in range)
setOffset(xOffset, yOffset);
}
void TiledImage::setOffset(int16_t x, int16_t y)
{
setXOffset(x);
setYOffset(y);
}
void TiledImage::setXOffset(int16_t x)
{
xOffset = x;
if (bitmap.getWidth() != 0)
{
xOffset = ((xOffset % bitmap.getWidth()) + bitmap.getWidth()) % bitmap.getWidth();
}
}
void TiledImage::setYOffset(int16_t y)
{
yOffset = y;
if (bitmap.getHeight() != 0)
{
yOffset = ((yOffset % bitmap.getHeight()) + bitmap.getHeight()) % bitmap.getHeight();
}
}
void TiledImage::getOffset(int16_t& x, int16_t& y)
{
x = getXOffset();
y = getYOffset();
}
int16_t TiledImage::getXOffset()
{
return xOffset;
}
int16_t TiledImage::getYOffset()
{
return yOffset;
}
void TiledImage::draw(const Rect& invalidatedArea) const
{
uint16_t bitmapWidth = bitmap.getWidth();
uint16_t bitmapHeight = bitmap.getHeight();
if (bitmapWidth == 0 || bitmapHeight == 0)
{
return;
}
Rect meAbs;
translateRectToAbsolute(meAbs);
const int16_t minX = ((invalidatedArea.x + xOffset) / bitmapWidth) * bitmapWidth - xOffset;
const int16_t maxX = (((invalidatedArea.right() + xOffset) - 1) / bitmapWidth) * bitmapWidth;
const int16_t minY = ((invalidatedArea.y + yOffset) / bitmapHeight) * bitmapHeight - yOffset;
const int16_t maxY = (((invalidatedArea.bottom() + yOffset) - 1) / bitmapHeight) * bitmapHeight;
for (int16_t x = minX; x <= maxX; x += bitmapWidth)
{
for (int16_t y = minY; y <= maxY; y += bitmapHeight)
{
Rect dirty = Rect(x, y, bitmapWidth, bitmapHeight) & invalidatedArea;
dirty.x -= x;
dirty.y -= y;
HAL::lcd().drawPartialBitmap(bitmap, meAbs.x + x, meAbs.y + y, dirty, alpha);
}
}
}
Rect TiledImage::getSolidRect() const
{
if (alpha < 255)
{
return Rect(0, 0, 0, 0);
}
Rect solidRect = bitmap.getSolidRect();
if (solidRect.width == bitmap.getWidth())
{
solidRect.width = getWidth();
}
else
{
solidRect.x -= xOffset;
Rect solidRect2 = solidRect;
solidRect2.x += bitmap.getWidth();
if (solidRect.x < 0)
{
int16_t right = solidRect.right();
solidRect.width = MAX(right, 0);
solidRect.x = 0;
}
if (solidRect2.right() > getWidth())
{
solidRect2.width = solidRect2.right() - getWidth();
}
if (solidRect2.width > solidRect.width)
{
solidRect = solidRect2;
}
}
if (solidRect.height == bitmap.getHeight())
{
solidRect.height = getHeight();
}
else
{
solidRect.y -= yOffset;
Rect solidRect2 = solidRect;
solidRect2.y += bitmap.getHeight();
if (solidRect.y < 0)
{
int16_t bottom = solidRect.bottom();
solidRect.height = MAX(bottom, 0);
solidRect.y = 0;
}
if (solidRect2.bottom() > getHeight())
{
solidRect2.height = solidRect2.bottom() - getHeight();
}
if (solidRect2.height > solidRect.height)
{
solidRect = solidRect2;
}
}
return solidRect;
}
} // namespace touchgfx

View File

@ -0,0 +1,52 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <touchgfx/widgets/ToggleButton.hpp>
namespace touchgfx
{
void ToggleButton::forceState(bool activeState)
{
if (activeState)
{
// up should equal originalPressed
if (up.getId() != originalPressed.getId())
{
down = up;
up = originalPressed;
}
}
else
{
// down should equal originalPressed
if (down.getId() != originalPressed.getId())
{
up = down;
down = originalPressed;
}
}
}
void ToggleButton::handleClickEvent(const ClickEvent& event)
{
if (pressed && (event.getType() == ClickEvent::RELEASED))
{
Bitmap tmp = up;
up = down;
down = tmp;
}
Button::handleClickEvent(event);
}
} // namespace touchgfx

View File

@ -0,0 +1,28 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <touchgfx/widgets/TouchArea.hpp>
namespace touchgfx
{
void TouchArea::handleClickEvent(const ClickEvent& event)
{
AbstractButton::handleClickEvent(event);
if (pressedAction && pressedAction->isValid() && pressed)
{
pressedAction->execute(*this);
}
}
} // namespace touchgfx

View File

@ -0,0 +1,69 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <touchgfx/Color.hpp>
#include <touchgfx/widgets/canvas/AbstractPainterABGR2222.hpp>
namespace touchgfx
{
void AbstractPainterABGR2222::render(uint8_t* ptr,
int x,
int xAdjust,
int y,
unsigned count,
const uint8_t* covers)
{
uint8_t* p = ptr + (x + xAdjust);
currentX = x + areaOffsetX;
currentY = y + areaOffsetY;
if (renderInit())
{
do
{
uint8_t red, green, blue, alpha;
if (renderNext(red, green, blue, alpha))
{
const uint8_t combinedAlpha = LCD::div255((*covers) * LCD::div255(alpha * widgetAlpha));
if (combinedAlpha == 0xFF) // max alpha=0xFF on "*covers" and max alpha=0xFF on "widgetAlpha"
{
// Render a solid pixel
renderPixel(p, red, green, blue);
}
else
{
const uint8_t ialpha = 0xFF - combinedAlpha;
const uint8_t p_red = LCD8bpp_ABGR2222::getRedFromColor(*p);
const uint8_t p_green = LCD8bpp_ABGR2222::getGreenFromColor(*p);
const uint8_t p_blue = LCD8bpp_ABGR2222::getBlueFromColor(*p);
renderPixel(p,
LCD::div255(red * combinedAlpha + p_red * ialpha),
LCD::div255(green * combinedAlpha + p_green * ialpha),
LCD::div255(blue * combinedAlpha + p_blue * ialpha));
}
}
covers++;
p++;
currentX++;
} while (--count != 0);
}
}
void AbstractPainterABGR2222::renderPixel(uint8_t* p, uint8_t red, uint8_t green, uint8_t blue)
{
*p = LCD8bpp_ABGR2222::getColorFromRGB(red, green, blue);
}
} // namespace touchgfx

View File

@ -0,0 +1,69 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <touchgfx/Color.hpp>
#include <touchgfx/widgets/canvas/AbstractPainterARGB2222.hpp>
namespace touchgfx
{
void AbstractPainterARGB2222::render(uint8_t* ptr,
int x,
int xAdjust,
int y,
unsigned count,
const uint8_t* covers)
{
uint8_t* p = ptr + (x + xAdjust);
currentX = x + areaOffsetX;
currentY = y + areaOffsetY;
if (renderInit())
{
do
{
uint8_t red, green, blue, alpha;
if (renderNext(red, green, blue, alpha))
{
const uint8_t combinedAlpha = LCD::div255((*covers) * LCD::div255(alpha * widgetAlpha));
if (combinedAlpha == 0xFF) // max alpha=255 on "*covers" and max alpha=255 on "widgetAlpha"
{
// Render a solid pixel
renderPixel(p, red, green, blue);
}
else
{
const uint8_t ialpha = 0xFF - combinedAlpha;
const uint8_t p_red = LCD8bpp_ARGB2222::getRedFromColor(*p);
const uint8_t p_green = LCD8bpp_ARGB2222::getGreenFromColor(*p);
const uint8_t p_blue = LCD8bpp_ARGB2222::getBlueFromColor(*p);
renderPixel(p,
LCD::div255(red * combinedAlpha + p_red * ialpha),
LCD::div255(green * combinedAlpha + p_green * ialpha),
LCD::div255(blue * combinedAlpha + p_blue * ialpha));
}
}
covers++;
p++;
currentX++;
} while (--count != 0);
}
}
void AbstractPainterARGB2222::renderPixel(uint8_t* p, uint8_t red, uint8_t green, uint8_t blue)
{
*p = LCD8bpp_ARGB2222::getColorFromRGB(red, green, blue);
}
} // namespace touchgfx

View File

@ -0,0 +1,80 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <touchgfx/Color.hpp>
#include <touchgfx/widgets/canvas/AbstractPainterARGB8888.hpp>
namespace touchgfx
{
void AbstractPainterARGB8888::render(uint8_t* ptr,
int x,
int xAdjust,
int y,
unsigned count,
const uint8_t* covers)
{
uint8_t* p = ptr + ((x + xAdjust) * 4);
currentX = x + areaOffsetX;
currentY = y + areaOffsetY;
if (renderInit())
{
do
{
uint8_t red, green, blue, alpha;
if (renderNext(red, green, blue, alpha))
{
const uint8_t combinedAlpha = LCD::div255((*covers) * LCD::div255(alpha * widgetAlpha));
if (combinedAlpha == 0xFF) // max alpha=0xFF on "*covers" and max alpha=0xFF on "widgetAlpha"
{
// Render a solid pixel
renderPixel(reinterpret_cast<uint16_t*>(p), red, green, blue);
}
else
{
const uint8_t ialpha = 0xFF - combinedAlpha;
const uint8_t p_blue = p[0];
const uint8_t p_green = p[1];
const uint8_t p_red = p[2];
const uint8_t p_alpha = p[3];
renderPixel(reinterpret_cast<uint16_t*>(p),
LCD::div255(red * combinedAlpha + p_red * ialpha),
LCD::div255(green * combinedAlpha + p_green * ialpha),
LCD::div255(blue * combinedAlpha + p_blue * ialpha),
p_alpha + combinedAlpha - LCD::div255(p_alpha * combinedAlpha));
}
}
covers++;
p += 4;
currentX++;
} while (--count != 0);
}
}
void AbstractPainterARGB8888::renderPixel(uint16_t* p, uint8_t red, uint8_t green, uint8_t blue)
{
renderPixel(p, red, green, blue, 0xFF);
}
void AbstractPainterARGB8888::renderPixel(uint16_t* p, uint8_t red, uint8_t green, uint8_t blue, uint8_t alpha)
{
uint8_t* p8 = reinterpret_cast<uint8_t*>(p);
p8[0] = blue;
p8[1] = green;
p8[2] = red;
p8[3] = alpha;
}
} // namespace touchgfx

View File

@ -0,0 +1,69 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <touchgfx/Color.hpp>
#include <touchgfx/widgets/canvas/AbstractPainterBGRA2222.hpp>
namespace touchgfx
{
void AbstractPainterBGRA2222::render(uint8_t* ptr,
int x,
int xAdjust,
int y,
unsigned count,
const uint8_t* covers)
{
uint8_t* p = ptr + (x + xAdjust);
currentX = x + areaOffsetX;
currentY = y + areaOffsetY;
if (renderInit())
{
do
{
uint8_t red, green, blue, alpha;
if (renderNext(red, green, blue, alpha))
{
const uint8_t combinedAlpha = LCD::div255((*covers) * LCD::div255(alpha * widgetAlpha));
if (combinedAlpha == 0xFF) // max alpha=255 on "*covers" and max alpha=255 on "widgetAlpha"
{
// Render a solid pixel
renderPixel(p, red, green, blue);
}
else
{
const uint8_t ialpha = 0xFF - combinedAlpha;
const uint8_t p_red = LCD8bpp_BGRA2222::getRedFromColor(*p);
const uint8_t p_green = LCD8bpp_BGRA2222::getGreenFromColor(*p);
const uint8_t p_blue = LCD8bpp_BGRA2222::getBlueFromColor(*p);
renderPixel(p,
LCD::div255(red * combinedAlpha + p_red * ialpha),
LCD::div255(green * combinedAlpha + p_green * ialpha),
LCD::div255(blue * combinedAlpha + p_blue * ialpha));
}
}
covers++;
p++;
currentX++;
} while (--count != 0);
}
}
void AbstractPainterBGRA2222::renderPixel(uint8_t* p, uint8_t red, uint8_t green, uint8_t blue)
{
*p = LCD8bpp_BGRA2222::getColorFromRGB(red, green, blue);
}
} // namespace touchgfx

View File

@ -0,0 +1,95 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <touchgfx/widgets/canvas/AbstractPainterBW.hpp>
namespace touchgfx
{
void AbstractPainterBW::render(uint8_t* ptr,
int x,
int xAdjust,
int y,
unsigned count,
const uint8_t* covers)
{
currentX = x + areaOffsetX;
currentY = y + areaOffsetY;
x += xAdjust;
unsigned char* p = ptr + (x / 8);
if (!renderInit())
{
return;
}
const uint8_t totalAlpha = LCD::div255(widgetAlpha * painterAlpha);
if (totalAlpha == 0xFF)
{
do
{
uint8_t color;
if (renderNext(color))
{
if (*covers >= 0x80)
{
unsigned pixel = 1 << (7 - (x % 8));
if (!color)
{
*p &= ~pixel;
}
else
{
*p |= pixel;
}
}
}
if (((++x) % 8) == 0)
{
p++;
}
covers++;
currentX++;
} while (--count);
}
else
{
do
{
uint8_t color;
if (renderNext(color))
{
if (totalAlpha * *covers >= 0xFF * 0x80)
{
unsigned pixel = 1 << (7 - (x % 8));
if (!color)
{
*p &= ~pixel;
}
else
{
*p |= pixel;
}
}
}
if (((++x) % 8) == 0)
{
p++;
}
covers++;
currentX++;
} while (--count);
}
}
} // namespace touchgfx

View File

@ -0,0 +1,64 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <platform/driver/lcd/LCD2bpp.hpp>
#include <touchgfx/Color.hpp>
#include <touchgfx/widgets/canvas/AbstractPainterGRAY2.hpp>
namespace touchgfx
{
void AbstractPainterGRAY2::render(uint8_t* ptr,
int x,
int xAdjust,
int y,
unsigned count,
const uint8_t* covers)
{
currentX = x + areaOffsetX;
currentY = y + areaOffsetY;
x += xAdjust;
if (renderInit())
{
do
{
uint8_t gray, alpha;
if (renderNext(gray, alpha))
{
const uint8_t combinedAlpha = LCD::div255((*covers) * LCD::div255(alpha * widgetAlpha));
if (combinedAlpha == 0xFF) // max alpha=0xFF on "*covers" and max alpha=0xFF on "widgetAlpha"
{
// Render a solid pixel
renderPixel(ptr, x, gray);
}
else
{
const uint8_t p_gray = LCD2bpp::getPixel(ptr, x) * 0x55;
const uint8_t ialpha = 0xFF - combinedAlpha;
renderPixel(ptr, x, LCD::div255((gray * combinedAlpha + p_gray * ialpha) * 0x55) >> 6);
}
}
covers++;
x++;
currentX++;
} while (--count != 0);
}
}
void AbstractPainterGRAY2::renderPixel(uint8_t* p, uint16_t offset, uint8_t gray)
{
LCD2bpp::setPixel(p, offset, gray);
}
} // namespace touchgfx

View File

@ -0,0 +1,64 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <platform/driver/lcd/LCD4bpp.hpp>
#include <touchgfx/Color.hpp>
#include <touchgfx/widgets/canvas/AbstractPainterGRAY4.hpp>
namespace touchgfx
{
void AbstractPainterGRAY4::render(uint8_t* ptr,
int x,
int xAdjust,
int y,
unsigned count,
const uint8_t* covers)
{
currentX = x + areaOffsetX;
currentY = y + areaOffsetY;
x += xAdjust;
if (renderInit())
{
do
{
uint8_t gray, alpha;
if (renderNext(gray, alpha))
{
const uint8_t combinedAlpha = LCD::div255((*covers) * LCD::div255(alpha * widgetAlpha));
if (combinedAlpha == 0xFF) // max alpha=0xFF on "*covers" and max alpha=0xFF on "widgetAlpha"
{
// Render a solid pixel
renderPixel(ptr, x, gray);
}
else
{
const uint8_t p_gray = LCD4bpp::getPixel(ptr, x);
const uint8_t ialpha = 0xFF - combinedAlpha;
renderPixel(ptr, x, LCD::div255((gray * combinedAlpha + p_gray * ialpha) * 0x11) >> 4);
}
}
covers++;
x++;
currentX++;
} while (--count != 0);
}
}
void AbstractPainterGRAY4::renderPixel(uint8_t* p, uint16_t offset, uint8_t gray)
{
LCD4bpp::setPixel(p, offset, gray & 0x0F);
}
} // namespace touchgfx

View File

@ -0,0 +1,69 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <touchgfx/Color.hpp>
#include <touchgfx/widgets/canvas/AbstractPainterRGB565.hpp>
namespace touchgfx
{
void AbstractPainterRGB565::render(uint8_t* ptr,
int x,
int xAdjust,
int y,
unsigned count,
const uint8_t* covers)
{
uint16_t* p = reinterpret_cast<uint16_t*>(ptr) + (x + xAdjust);
currentX = x + areaOffsetX;
currentY = y + areaOffsetY;
if (renderInit())
{
do
{
uint8_t red, green, blue, alpha;
if (renderNext(red, green, blue, alpha))
{
const uint8_t combinedAlpha = LCD::div255((*covers) * LCD::div255(alpha * widgetAlpha));
if (combinedAlpha == 0xFF) // max alpha=0xFF on "*covers" and max alpha=0xFF on "widgetAlpha"
{
// Render a solid pixel
renderPixel(p, red, green, blue);
}
else
{
const uint8_t ialpha = 0xFF - combinedAlpha;
const uint8_t p_red = (*p & RMASK) >> 8;
const uint8_t p_green = (*p & GMASK) >> 3;
const uint8_t p_blue = (*p & BMASK) << 3;
renderPixel(p,
LCD::div255(red * combinedAlpha + p_red * ialpha),
LCD::div255(green * combinedAlpha + p_green * ialpha),
LCD::div255(blue * combinedAlpha + p_blue * ialpha));
}
}
covers++;
p++;
currentX++;
} while (--count != 0);
}
}
void AbstractPainterRGB565::renderPixel(uint16_t* p, uint8_t red, uint8_t green, uint8_t blue)
{
*p = ((red << 8) & RMASK) | ((green << 3) & GMASK) | ((blue >> 3) & BMASK);
}
} // namespace touchgfx

View File

@ -0,0 +1,72 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <touchgfx/Color.hpp>
#include <touchgfx/widgets/canvas/AbstractPainterRGB888.hpp>
namespace touchgfx
{
void AbstractPainterRGB888::render(uint8_t* ptr,
int x,
int xAdjust,
int y,
unsigned count,
const uint8_t* covers)
{
uint8_t* p = ptr + ((x + xAdjust) * 3);
currentX = x + areaOffsetX;
currentY = y + areaOffsetY;
if (renderInit())
{
do
{
uint8_t red, green, blue, alpha;
if (renderNext(red, green, blue, alpha))
{
const uint8_t combinedAlpha = LCD::div255((*covers) * LCD::div255(alpha * widgetAlpha));
if (combinedAlpha == 0xFF) // max alpha=0xFF on "*covers" and max alpha=0xFF on "widgetAlpha"
{
// Render a solid pixel
renderPixel(reinterpret_cast<uint16_t*>(p), red, green, blue);
}
else
{
const uint8_t ialpha = 0xFF - combinedAlpha;
const uint8_t p_blue = p[0];
const uint8_t p_green = p[1];
const uint8_t p_red = p[2];
renderPixel(reinterpret_cast<uint16_t*>(p),
LCD::div255(red * combinedAlpha + p_red * ialpha),
LCD::div255(green * combinedAlpha + p_green * ialpha),
LCD::div255(blue * combinedAlpha + p_blue * ialpha));
}
}
covers++;
p += 3;
currentX++;
} while (--count != 0);
}
}
void AbstractPainterRGB888::renderPixel(uint16_t* p, uint8_t red, uint8_t green, uint8_t blue)
{
uint8_t* p8 = reinterpret_cast<uint8_t*>(p);
p8[0] = blue;
p8[1] = green;
p8[2] = red;
}
} // namespace touchgfx

View File

@ -0,0 +1,69 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <touchgfx/Color.hpp>
#include <touchgfx/widgets/canvas/AbstractPainterRGBA2222.hpp>
namespace touchgfx
{
void AbstractPainterRGBA2222::render(uint8_t* ptr,
int x,
int xAdjust,
int y,
unsigned count,
const uint8_t* covers)
{
uint8_t* p = ptr + (x + xAdjust);
currentX = x + areaOffsetX;
currentY = y + areaOffsetY;
if (renderInit())
{
do
{
uint8_t red, green, blue, alpha;
if (renderNext(red, green, blue, alpha))
{
const uint8_t combinedAlpha = LCD::div255((*covers) * LCD::div255(alpha * widgetAlpha));
if (combinedAlpha == 0xFF) // max alpha=0xFF on "*covers" and max alpha=0xFF on "widgetAlpha"
{
// Render a solid pixel
renderPixel(p, red, green, blue);
}
else
{
const uint8_t ialpha = 0xFF - combinedAlpha;
const uint8_t p_red = LCD8bpp_RGBA2222::getRedFromColor(*p);
const uint8_t p_green = LCD8bpp_RGBA2222::getGreenFromColor(*p);
const uint8_t p_blue = LCD8bpp_RGBA2222::getBlueFromColor(*p);
renderPixel(p,
LCD::div255(red * combinedAlpha + p_red * ialpha),
LCD::div255(green * combinedAlpha + p_green * ialpha),
LCD::div255(blue * combinedAlpha + p_blue * ialpha));
}
}
covers++;
p++;
currentX++;
} while (--count != 0);
}
}
void AbstractPainterRGBA2222::renderPixel(uint8_t* p, uint8_t red, uint8_t green, uint8_t blue)
{
*p = LCD8bpp_RGBA2222::getColorFromRGB(red, green, blue);
}
} // namespace touchgfx

View File

@ -0,0 +1,88 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <touchgfx/widgets/canvas/AbstractShape.hpp>
#include <touchgfx/widgets/canvas/Canvas.hpp>
namespace touchgfx
{
AbstractShape::AbstractShape()
: CanvasWidget(),
dx(0), dy(0), shapeAngle(0),
xScale(CWRUtil::toQ10<int>(1)), yScale(CWRUtil::toQ10<int>(1)),
minimalRect(Rect())
{
Drawable::setWidthHeight(0, 0);
}
bool AbstractShape::drawCanvasWidget(const Rect& invalidatedArea) const
{
Canvas canvas(this, invalidatedArea);
int numPoints = getNumPoints();
if (numPoints == 0)
{
return true;
}
canvas.moveTo(getCacheX(0), getCacheY(0));
for (int i = 1; i < numPoints; i++)
{
canvas.lineTo(getCacheX(i), getCacheY(i));
}
return canvas.render();
}
void AbstractShape::updateAbstractShapeCache()
{
int numPoints = getNumPoints();
int xMin = 0;
int xMax = 0;
int yMin = 0;
int yMax = 0;
for (int i = 0; i < numPoints; i++)
{
CWRUtil::Q5 xCorner = getCornerX(i);
CWRUtil::Q5 yCorner = getCornerY(i);
CWRUtil::Q5 xCache = dx + ((CWRUtil::mulQ5(xCorner, xScale) * CWRUtil::cosine(shapeAngle))) - ((CWRUtil::mulQ5(yCorner, yScale) * CWRUtil::sine(shapeAngle)));
if (i == 0 || xCache.to<int>() > xMax)
{
xMax = xCache.to<int>();
}
if (i == 0 || xCache.to<int>() < xMin)
{
xMin = xCache.to<int>();
}
CWRUtil::Q5 yCache = dy + ((CWRUtil::mulQ5(yCorner, yScale) * CWRUtil::cosine(shapeAngle))) + ((CWRUtil::mulQ5(xCorner, xScale) * CWRUtil::sine(shapeAngle)));
if (i == 0 || yCache.to<int>() > yMax)
{
yMax = yCache.to<int>();
}
if (i == 0 || yCache.to<int>() < yMin)
{
yMin = yCache.to<int>();
}
setCache(i, xCache, yCache);
}
minimalRect = Rect(xMin, yMin, xMax - xMin + 1, yMax - yMin + 1);
}
Rect AbstractShape::getMinimalRect() const
{
return minimalRect;
}
} // namespace touchgfx

View File

@ -0,0 +1,277 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <touchgfx/widgets/canvas/Canvas.hpp>
namespace touchgfx
{
Canvas::Canvas(const CanvasWidget* _widget, const Rect& invalidatedArea)
: widget(_widget),
enoughMemory(false), penUp(true), penHasBeenDown(false), previousOutside(0), penDownOutside(0)
{
assert(CanvasWidgetRenderer::hasBuffer() && "No buffer allocated for CanvasWidgetRenderer drawing");
assert(Rasterizer::POLY_BASE_SHIFT == 5 && "CanvasWidget assumes Q5 but Rasterizer uses a different setting");
// Area to redraw (relative coordinates)
Rect dirtyArea = Rect(0, 0, widget->getWidth(), widget->getHeight()) & invalidatedArea;
// Absolute position of the scalableImage.
Rect dirtyAreaAbsolute = dirtyArea;
widget->translateRectToAbsolute(dirtyAreaAbsolute);
// Transform rects to match framebuffer coordinates
// This is needed if the display is rotated compared to the framebuffer
DisplayTransformation::transformDisplayToFrameBuffer(dirtyArea, widget->getRect());
DisplayTransformation::transformDisplayToFrameBuffer(dirtyAreaAbsolute);
// Re-size buffers for optimum memory buffer layout.
enoughMemory = CanvasWidgetRenderer::setScanlineWidth(dirtyArea.width);
ras.reset();
offsetX = dirtyArea.x;
offsetY = dirtyArea.y;
invalidatedAreaX = CWRUtil::toQ5<int>(dirtyArea.x);
invalidatedAreaY = CWRUtil::toQ5<int>(dirtyArea.y);
invalidatedAreaWidth = CWRUtil::toQ5<int>(dirtyArea.width);
invalidatedAreaHeight = CWRUtil::toQ5<int>(dirtyArea.height);
// Create the rendering buffer
uint8_t* RESTRICT buf = reinterpret_cast<uint8_t*>(HAL::getInstance()->lockFrameBuffer());
int stride = HAL::lcd().framebufferStride();
uint8_t offset = 0;
switch (HAL::lcd().framebufferFormat())
{
case Bitmap::BW:
buf += (dirtyAreaAbsolute.x / 8) + dirtyAreaAbsolute.y * stride;
offset = dirtyAreaAbsolute.x % 8;
break;
case Bitmap::GRAY2:
buf += (dirtyAreaAbsolute.x / 4) + dirtyAreaAbsolute.y * stride;
offset = dirtyAreaAbsolute.x % 4;
break;
case Bitmap::GRAY4:
buf += (dirtyAreaAbsolute.x / 2) + dirtyAreaAbsolute.y * stride;
offset = dirtyAreaAbsolute.x % 2;
break;
case Bitmap::RGB565:
buf += dirtyAreaAbsolute.x * 2 + dirtyAreaAbsolute.y * stride;
break;
case Bitmap::RGB888:
buf += dirtyAreaAbsolute.x * 3 + dirtyAreaAbsolute.y * stride;
break;
case Bitmap::RGBA2222:
case Bitmap::BGRA2222:
case Bitmap::ARGB2222:
case Bitmap::ABGR2222:
case Bitmap::L8:
buf += dirtyAreaAbsolute.x + dirtyAreaAbsolute.y * stride;
break;
case Bitmap::ARGB8888:
buf += dirtyAreaAbsolute.x * 4 + dirtyAreaAbsolute.y * stride;
break;
case Bitmap::BW_RLE:
case Bitmap::A4:
case Bitmap::CUSTOM:
assert(0 && "Unsupported bit depth");
break;
}
ras.setMaxRenderY(dirtyAreaAbsolute.height);
rbuf.attach(buf, offset, dirtyAreaAbsolute.width, dirtyAreaAbsolute.height, stride);
}
Canvas::~Canvas()
{
HAL::getInstance()->unlockFrameBuffer(); //lint !e1551
}
void Canvas::moveTo(CWRUtil::Q5 x, CWRUtil::Q5 y)
{
if (!enoughMemory)
{
return;
}
if (!penUp)
{
close();
}
transformFrameBufferToDisplay(x, y);
x = x - invalidatedAreaX;
y = y - invalidatedAreaY;
uint8_t outside = isOutside(x, y, invalidatedAreaWidth, invalidatedAreaHeight);
if (outside)
{
penUp = true;
}
else
{
penDownOutside = outside;
ras.moveTo(x, y);
penUp = false;
penHasBeenDown = true;
}
initialX = x;
initialY = y;
previousX = x;
previousY = y;
previousOutside = outside;
}
void Canvas::lineTo(CWRUtil::Q5 x, CWRUtil::Q5 y)
{
if (!enoughMemory)
{
return;
}
transformFrameBufferToDisplay(x, y);
x = x - invalidatedAreaX;
y = y - invalidatedAreaY;
uint8_t outside = isOutside(x, y, invalidatedAreaWidth, invalidatedAreaHeight);
if (!previousOutside)
{
ras.lineTo(x, y);
}
else
{
if (!outside || !(previousOutside & outside))
{
// x,y is inside, or on another side compared to previous
if (penUp)
{
penDownOutside = previousOutside;
ras.moveTo(previousX, previousY);
penUp = false;
penHasBeenDown = true;
}
else
{
ras.lineTo(previousX, previousY);
}
ras.lineTo(x, y);
}
else
{
// Restrict "outside" to the same side as previous point.
outside &= previousOutside;
}
}
previousX = x;
previousY = y;
previousOutside = outside;
}
bool Canvas::render(uint8_t customAlpha)
{
// If the invalidated rect is too wide compared to the allocated buffer for CWR,
// redrawing will not help. The CanvasWidget needs to know about this situation
// and maybe try to divide the area vertically instead, but this has not been
// implemented. And probably should not.
if (!enoughMemory)
{
return true; // Redrawing a rect with fewer scanlines will not help, fake "ok" to move on
}
if (ras.wasOutlineTooComplex())
{
return false; // Try again with fewer scanlines
}
if (!penHasBeenDown)
{
return true; // Nothing drawn. Done
}
const uint8_t alpha = LCD::div255(widget->getAlpha() * customAlpha);
if (alpha == 0)
{
return true; // Invisible. Done
}
close();
widget->getPainter().setOffset(offsetX /*+widget->getX()*/, offsetY /*+widget->getY()*/);
widget->getPainter().setWidgetAlpha(alpha);
Renderer renderer(rbuf, widget->getPainter());
return ras.render(renderer);
}
uint8_t Canvas::isOutside(const CWRUtil::Q5& x, const CWRUtil::Q5& y, const CWRUtil::Q5& width, const CWRUtil::Q5& height) const
{
uint8_t outside = 0;
// Find out if (x,y) is above/below of current area
if (y < 0)
{
outside = POINT_IS_ABOVE;
}
else if (y >= height)
{
outside = POINT_IS_BELOW;
}
// Find out if (x,y) is left/right of current area
if (x < 0)
{
outside |= POINT_IS_LEFT;
}
else if (x >= width)
{
outside |= POINT_IS_RIGHT;
}
return outside;
}
void Canvas::transformFrameBufferToDisplay(CWRUtil::Q5& x, CWRUtil::Q5& y) const
{
switch (HAL::DISPLAY_ROTATION)
{
case rotate0:
break;
case rotate90:
CWRUtil::Q5 tmpY = y;
y = CWRUtil::toQ5<int>(widget->getWidth()) - x;
x = tmpY;
break;
}
}
void Canvas::close()
{
if (!penUp)
{
if (previousOutside & penDownOutside)
{
// We are outside on the same side as we started. No need
// to close the path, CWR will do this for us.
//lineTo(penDownX, penDownY);
}
else
{
if (previousOutside)
{
ras.lineTo(previousX, previousY);
}
ras.lineTo(initialX, initialY);
}
}
penUp = false;
}
} // namespace touchgfx

View File

@ -0,0 +1,146 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <touchgfx/widgets/canvas/CanvasWidget.hpp>
namespace touchgfx
{
CanvasWidget::CanvasWidget()
: Widget(),
canvasPainter(0),
maxRenderLines(0x7FFF),
alpha(255)
{
}
void CanvasWidget::setPainter(AbstractPainter& painter)
{
canvasPainter = &painter;
}
AbstractPainter& CanvasWidget::getPainter() const
{
assert(canvasPainter != 0 && "No painter set");
return *canvasPainter; //lint !e613
} //lint !e1763
void CanvasWidget::draw(const Rect& invalidatedArea) const
{
Rect area = invalidatedArea;
int16_t* offset;
int16_t* lines;
int16_t* width;
int16_t* height;
int16_t wantedRenderLines;
switch (HAL::DISPLAY_ROTATION)
{
default:
case rotate0:
offset = &area.y;
lines = &area.height;
width = &area.width;
height = &wantedRenderLines;
break;
case rotate90:
offset = &area.x;
lines = &area.width;
width = &wantedRenderLines;
height = &area.height;
break;
}
Rect minimalRect = getMinimalRect();
bool failedAtLeastOnce = false;
while (*lines)
{
wantedRenderLines = MIN(maxRenderLines, *lines);
while (wantedRenderLines > 0)
{
Rect smallArea(area.x, area.y, *width, *height);
if (!smallArea.intersect(minimalRect))
{
break;
}
if (drawCanvasWidget(smallArea))
{
break;
}
#ifdef SIMULATOR
if (CanvasWidgetRenderer::getWriteMemoryUsageReport())
{
if (wantedRenderLines > 1)
{
touchgfx_printf("CWR will split draw into multiple draws due to limited memory.\n");
}
else
{
touchgfx_printf("CWR was unable to complete a draw operation due to limited memory.\n");
}
}
#endif
wantedRenderLines >>= 1;
failedAtLeastOnce = true;
}
if (wantedRenderLines == 0)
{
// We did not manage to draw anything. Set wantedHeight to
// one to skip a single raster line and try to render the
// rest of the CanvasWidget.
wantedRenderLines = 1;
}
else
{
if (failedAtLeastOnce && maxRenderLines == 0x7FFF)
{
// Only adjust maxRenderLines if it is the first draw for the CanvasWidget
maxRenderLines = wantedRenderLines;
}
}
*offset += wantedRenderLines;
*lines -= wantedRenderLines;
}
if (maxRenderLines == 0x7FFF)
{
maxRenderLines--; // 0x7FFF means first draw
}
}
void CanvasWidget::invalidate() const
{
Rect minimalRect = getMinimalRect();
minimalRect.intersect(CanvasWidget::getMinimalRect());
invalidateRect(minimalRect);
}
Rect CanvasWidget::getMinimalRect() const
{
return Rect(0, 0, getWidth(), getHeight());
}
Rect CanvasWidget::getSolidRect() const
{
return Rect(0, 0, 0, 0);
}
void CanvasWidget::resetMaxRenderLines()
{
maxRenderLines = 0x7FFF;
}
} // namespace touchgfx

View File

@ -0,0 +1,554 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <touchgfx/widgets/canvas/Circle.hpp>
namespace touchgfx
{
Circle::Circle()
: CanvasWidget(),
circleCenterX(0), circleCenterY(0), circleRadius(0),
circleArcAngleStart(CWRUtil::toQ5<int>(0)), circleArcAngleEnd(CWRUtil::toQ5<int>(360)),
circleLineWidth(0), circleArcIncrement(5),
circleCapArcIncrement(180)
{
Drawable::setWidthHeight(0, 0);
}
void Circle::setPrecision(int precision)
{
if (precision < 1)
{
precision = 1;
}
if (precision > 120)
{
precision = 120;
}
circleArcIncrement = precision;
}
int Circle::getPrecision() const
{
return circleArcIncrement;
}
void Circle::setCapPrecision(int precision)
{
if (precision < 1)
{
precision = 1;
}
if (precision > 180)
{
precision = 180;
}
circleCapArcIncrement = precision;
}
int Circle::getCapPrecision() const
{
return circleCapArcIncrement;
}
bool Circle::drawCanvasWidget(const Rect& invalidatedArea) const
{
CWRUtil::Q5 arcStart = circleArcAngleStart;
CWRUtil::Q5 arcEnd = circleArcAngleEnd;
CWRUtil::Q5 _360 = CWRUtil::toQ5<int>(360);
// Put start before end by swapping
if (arcStart > arcEnd)
{
CWRUtil::Q5 tmp = arcStart;
arcStart = arcEnd;
arcEnd = tmp;
}
if ((arcEnd - arcStart) >= _360)
{
// The entire circle has to be drawn
arcStart = CWRUtil::toQ5<int>(0);
arcEnd = _360;
}
if (circleLineWidth != 0)
{
// Check if invalidated area is completely inside the circle
int32_t x1 = int(CWRUtil::toQ5(invalidatedArea.x)); // Take the corners of the invalidated area
int32_t x2 = int(CWRUtil::toQ5(invalidatedArea.right()));
int32_t y1 = int(CWRUtil::toQ5(invalidatedArea.y));
int32_t y2 = int(CWRUtil::toQ5(invalidatedArea.bottom()));
int32_t dx1 = abs(int(circleCenterX) - x1); // Find distances between each corner and circle center
int32_t dx2 = abs(int(circleCenterX) - x2);
int32_t dy1 = abs(int(circleCenterY) - y1);
int32_t dy2 = abs(int(circleCenterY) - y2);
int32_t dx = CWRUtil::Q5(MAX(dx1, dx2)).to<int>() + 1; // Largest hor/vert distance (round up)
int32_t dy = CWRUtil::Q5(MAX(dy1, dy2)).to<int>() + 1;
int32_t dsqr = (dx * dx) + (dy * dy); // Pythagoras
// From https://www.mathopenref.com/polygonincircle.html
int32_t rmin = ((circleRadius - (circleLineWidth / 2)) * CWRUtil::cosine((circleArcIncrement + 1) / 2)).to<int>();
// Check if invalidatedArea is completely inside circle
if (dsqr < rmin * rmin)
{
return true;
}
}
Canvas canvas(this, invalidatedArea);
CWRUtil::Q5 radius = circleRadius;
CWRUtil::Q5 lineWidth = circleLineWidth;
if (circleLineWidth > circleRadius * 2)
{
lineWidth = (circleRadius + circleLineWidth / 2);
radius = lineWidth / 2;
}
CWRUtil::Q5 arc = arcStart;
CWRUtil::Q5 circleArcIncrementQ5 = CWRUtil::toQ5<int>(circleArcIncrement);
moveToAR2(canvas, arc, (radius * 2) + lineWidth);
CWRUtil::Q5 nextArc = CWRUtil::Q5(ROUNDUP((int)(arc + CWRUtil::toQ5<int>(1)), (int)circleArcIncrementQ5));
while (nextArc <= arcEnd)
{
arc = nextArc;
lineToAR2(canvas, arc, (radius * 2) + lineWidth);
nextArc = nextArc + circleArcIncrementQ5;
}
if (arc < arcEnd)
{
// "arc" is not updated. It is the last arc in steps of "circleArcIncrement"
lineToAR2(canvas, arcEnd, (radius * 2) + lineWidth);
}
if (lineWidth == CWRUtil::toQ5<int>(0))
{
// Draw a filled circle / pie / pacman
if (arcEnd - arcStart < _360)
{
// Not a complete circle, line to center
canvas.lineTo(circleCenterX, circleCenterY);
}
}
else
{
CWRUtil::Q5 circleCapArcIncrementQ5 = CWRUtil::toQ5<int>(circleCapArcIncrement);
CWRUtil::Q5 _180 = CWRUtil::toQ5<int>(180);
if (arcEnd - arcStart < _360)
{
// Draw the circle cap
CWRUtil::Q5 capX = circleCenterX + (radius * CWRUtil::sine(arcEnd));
CWRUtil::Q5 capY = circleCenterY - (radius * CWRUtil::cosine(arcEnd));
for (CWRUtil::Q5 capAngle = arcEnd + circleCapArcIncrementQ5; capAngle < arcEnd + _180; capAngle = capAngle + circleCapArcIncrementQ5)
{
lineToXYAR2(canvas, capX, capY, capAngle, lineWidth);
}
}
// Not a filled circle, draw the path on the inside of the circle
if (arc < arcEnd)
{
lineToAR2(canvas, arcEnd, (radius * 2) - lineWidth);
}
nextArc = arc;
while (nextArc >= arcStart)
{
arc = nextArc;
lineToAR2(canvas, arc, (radius * 2) - lineWidth);
nextArc = nextArc - circleArcIncrementQ5;
}
if (arc > arcStart)
{
lineToAR2(canvas, arcStart, (radius * 2) - lineWidth);
}
if (arcEnd - arcStart < _360)
{
// Draw the circle cap
CWRUtil::Q5 capX = circleCenterX + (radius * CWRUtil::sine(arcStart));
CWRUtil::Q5 capY = circleCenterY - (radius * CWRUtil::cosine(arcStart));
for (CWRUtil::Q5 capAngle = arcStart - _180 + circleCapArcIncrementQ5; capAngle < arcStart; capAngle = capAngle + circleCapArcIncrementQ5)
{
lineToXYAR2(canvas, capX, capY, capAngle, lineWidth);
}
}
}
return canvas.render();
}
Rect Circle::getMinimalRect() const
{
return getMinimalRect(circleArcAngleStart, circleArcAngleEnd);
}
Rect Circle::getMinimalRect(int16_t arcStart, int16_t arcEnd) const
{
return getMinimalRect(CWRUtil::toQ5<int>(arcStart), CWRUtil::toQ5<int>(arcEnd));
}
Rect Circle::getMinimalRect(CWRUtil::Q5 arcStart, CWRUtil::Q5 arcEnd) const
{
CWRUtil::Q5 xMin = CWRUtil::toQ5<int>(getWidth());
CWRUtil::Q5 xMax = CWRUtil::toQ5<int>(0);
CWRUtil::Q5 yMin = CWRUtil::toQ5<int>(getHeight());
CWRUtil::Q5 yMax = CWRUtil::toQ5<int>(0);
calculateMinimalRect(arcStart, arcEnd, xMin, xMax, yMin, yMax);
return Rect(xMin.to<int>() - 1, yMin.to<int>() - 1,
xMax.to<int>() - xMin.to<int>() + 2, yMax.to<int>() - yMin.to<int>() + 2);
}
void Circle::updateArc(CWRUtil::Q5 setStartAngleQ5, CWRUtil::Q5 setEndAngleQ5)
{
CWRUtil::Q5 startAngleQ5 = setStartAngleQ5;
CWRUtil::Q5 endAngleQ5 = setEndAngleQ5;
if (circleArcAngleStart == startAngleQ5 && circleArcAngleEnd == endAngleQ5)
{
return;
}
// Make sure old start < end
if (circleArcAngleStart > circleArcAngleEnd)
{
CWRUtil::Q5 tmp = circleArcAngleStart;
circleArcAngleStart = circleArcAngleEnd;
circleArcAngleEnd = tmp;
}
// Make sure new start < end
if (startAngleQ5 > endAngleQ5)
{
CWRUtil::Q5 tmp = startAngleQ5;
startAngleQ5 = endAngleQ5;
endAngleQ5 = tmp;
}
// Nice constant
const CWRUtil::Q5 _360 = CWRUtil::toQ5<int>(360);
// Get old circle range start in [0..360[
if (circleArcAngleStart >= _360)
{
int x = (circleArcAngleStart / _360).to<int>();
circleArcAngleStart = circleArcAngleStart - _360 * x;
circleArcAngleEnd = circleArcAngleEnd - _360 * x;
}
else if (circleArcAngleStart < 0)
{
int x = 1 + ((-circleArcAngleStart) / _360).to<int>();
circleArcAngleStart = circleArcAngleStart + _360 * x;
circleArcAngleEnd = circleArcAngleEnd + _360 * x;
}
// Detect full circle
if ((circleArcAngleEnd - circleArcAngleStart) > _360)
{
circleArcAngleEnd = circleArcAngleStart + _360;
}
// Get new circle range start in [0..360[
if (startAngleQ5 >= _360)
{
int x = (startAngleQ5 / _360).to<int>();
startAngleQ5 = startAngleQ5 - _360 * x;
endAngleQ5 = endAngleQ5 - _360 * x;
}
else if (startAngleQ5 < 0)
{
int x = 1 + (-startAngleQ5 / _360).to<int>();
startAngleQ5 = startAngleQ5 + _360 * x;
endAngleQ5 = endAngleQ5 + _360 * x;
}
// Detect full circle
if ((endAngleQ5 - startAngleQ5) >= _360)
{
// Align full new circle with old start.
// So old[90..270] -> new[0..360] becomes new[90..450] for smaller invalidated area
startAngleQ5 = circleArcAngleStart;
endAngleQ5 = startAngleQ5 + _360;
}
else if ((circleArcAngleEnd - circleArcAngleStart) >= _360)
{
// New circle is not full, but old is. Align old circle with new.
// So old[0..360] -> new[90..270] becomes old[90..450] for smaller invalidated area
circleArcAngleStart = startAngleQ5;
circleArcAngleEnd = circleArcAngleStart + _360;
}
// New start is after old end. Could be overlap
// if old[10..30]->new[350..380] becomes new[-10..20]
if (startAngleQ5 > circleArcAngleEnd && endAngleQ5 - _360 >= circleArcAngleStart)
{
startAngleQ5 = startAngleQ5 - _360;
endAngleQ5 = endAngleQ5 - _360;
}
// Same as above but for old instead of new
if (circleArcAngleStart > endAngleQ5 && circleArcAngleEnd - _360 >= startAngleQ5)
{
circleArcAngleStart = circleArcAngleStart - _360;
circleArcAngleEnd = circleArcAngleEnd - _360;
}
Rect r;
if (startAngleQ5 > circleArcAngleEnd || endAngleQ5 < circleArcAngleStart)
{
// Arcs do not overlap. Invalidate both arcs.
r = getMinimalRect(circleArcAngleStart, circleArcAngleEnd);
invalidateRect(r);
r = getMinimalRect(startAngleQ5, endAngleQ5);
invalidateRect(r);
}
else
{
// Arcs overlap. Invalidate both ends.
if (circleArcAngleStart != startAngleQ5)
{
r = getMinimalRectForUpdatedStartAngle(startAngleQ5);
invalidateRect(r);
}
if (circleArcAngleEnd != endAngleQ5)
{
r = getMinimalRectForUpdatedEndAngle(endAngleQ5);
invalidateRect(r);
}
}
circleArcAngleStart = setStartAngleQ5;
circleArcAngleEnd = setEndAngleQ5;
}
void Circle::moveToAR2(Canvas& canvas, const CWRUtil::Q5& angle, const CWRUtil::Q5& r2) const
{
canvas.moveTo(circleCenterX + ((r2 * CWRUtil::sine(angle)) / 2), circleCenterY - ((r2 * CWRUtil::cosine(angle)) / 2));
}
void Circle::lineToAR2(Canvas& canvas, const CWRUtil::Q5& angle, const CWRUtil::Q5& r2) const
{
lineToXYAR2(canvas, circleCenterX, circleCenterY, angle, r2);
}
void Circle::lineToXYAR2(Canvas& canvas, const CWRUtil::Q5& x, const CWRUtil::Q5& y, const CWRUtil::Q5& angle, const CWRUtil::Q5& r2) const
{
canvas.lineTo(x + ((r2 * CWRUtil::sine(angle)) / 2), y - ((r2 * CWRUtil::cosine(angle)) / 2));
}
void Circle::updateMinMaxAR(const CWRUtil::Q5& a, const CWRUtil::Q5& r2, CWRUtil::Q5& xMin, CWRUtil::Q5& xMax, CWRUtil::Q5& yMin, CWRUtil::Q5& yMax) const
{
CWRUtil::Q5 xNew = circleCenterX + ((r2 * CWRUtil::sine(a)) / 2);
CWRUtil::Q5 yNew = circleCenterY - ((r2 * CWRUtil::cosine(a)) / 2);
updateMinMaxXY(xNew, yNew, xMin, xMax, yMin, yMax);
}
void Circle::updateMinMaxXY(const CWRUtil::Q5& xNew, const CWRUtil::Q5& yNew, CWRUtil::Q5& xMin, CWRUtil::Q5& xMax, CWRUtil::Q5& yMin, CWRUtil::Q5& yMax) const
{
if (xNew < xMin)
{
xMin = xNew;
}
if (xNew > xMax)
{
xMax = xNew;
}
if (yNew < yMin)
{
yMin = yNew;
}
if (yNew > yMax)
{
yMax = yNew;
}
}
void Circle::calculateMinimalRect(CWRUtil::Q5 arcStart, CWRUtil::Q5 arcEnd, CWRUtil::Q5& xMin, CWRUtil::Q5& xMax, CWRUtil::Q5& yMin, CWRUtil::Q5& yMax) const
{
// Put start before end by swapping
if (arcStart > arcEnd)
{
CWRUtil::Q5 tmp = arcStart;
arcStart = arcEnd;
arcEnd = tmp;
}
CWRUtil::Q5 _90 = CWRUtil::toQ5<int>(90);
CWRUtil::Q5 _360 = CWRUtil::toQ5<int>(360);
if ((arcEnd - arcStart) >= _360)
{
// The entire circle has to be drawn
arcStart = CWRUtil::toQ5<int>(0);
arcEnd = _360;
}
// Check start angle
updateMinMaxAR(arcStart, (circleRadius * 2) + circleLineWidth, xMin, xMax, yMin, yMax);
// Here we have a up to 4 approximation steps on angles divisible by 90
CWRUtil::Q5 i;
for (i = CWRUtil::Q5(ROUNDUP((int)(arcStart + CWRUtil::toQ5<int>(1)), (int)_90)); i <= arcEnd; i = i + _90)
{
updateMinMaxAR(i, (circleRadius * 2) + circleLineWidth, xMin, xMax, yMin, yMax);
}
// Check end angle
if ((i - _90) < arcEnd)
{
updateMinMaxAR(arcEnd, (circleRadius * 2) + circleLineWidth, xMin, xMax, yMin, yMax);
}
if (circleLineWidth == CWRUtil::toQ5<int>(0))
{
// A filled circle / pie / pacman
if ((arcEnd - arcStart) < _360)
{
// Not a complete circle, check center
updateMinMaxAR(CWRUtil::toQ5<int>(0), CWRUtil::toQ5<int>(0), xMin, xMax, yMin, yMax);
}
}
else
{
// Not a filled circle, check the inside of the circle. Only start and/or end can cause new min/max values
updateMinMaxAR(arcStart, (circleRadius * 2) - circleLineWidth, xMin, xMax, yMin, yMax);
updateMinMaxAR(arcEnd, (circleRadius * 2) - circleLineWidth, xMin, xMax, yMin, yMax);
}
// Check if circle cap extends the min/max further
if ((circleCapArcIncrement < 180) && (arcEnd - arcStart < _360))
{
// Round caps
CWRUtil::Q5 capX = circleCenterX + (circleRadius * CWRUtil::sine(arcStart));
CWRUtil::Q5 capY = circleCenterY - (circleRadius * CWRUtil::cosine(arcStart));
updateMinMaxXY(capX - (circleLineWidth / 2), capY - (circleLineWidth / 2), xMin, xMax, yMin, yMax);
updateMinMaxXY(capX + (circleLineWidth / 2), capY + (circleLineWidth / 2), xMin, xMax, yMin, yMax);
capX = circleCenterX + (circleRadius * CWRUtil::sine(arcEnd));
capY = circleCenterY - (circleRadius * CWRUtil::cosine(arcEnd));
updateMinMaxXY(capX - (circleLineWidth / 2), capY - (circleLineWidth / 2), xMin, xMax, yMin, yMax);
updateMinMaxXY(capX + (circleLineWidth / 2), capY + (circleLineWidth / 2), xMin, xMax, yMin, yMax);
}
}
Rect Circle::getMinimalRectForUpdatedStartAngle(const CWRUtil::Q5& startAngleQ5) const
{
CWRUtil::Q5 minAngle = CWRUtil::Q5(0); // Unused default value
CWRUtil::Q5 maxAngle = CWRUtil::Q5(0); // Unused default value
int circleArcIncrementQ5int = (int)CWRUtil::toQ5<int>(circleArcIncrement);
if (circleArcAngleStart < circleArcAngleEnd)
{
// start is smaller than end
if (startAngleQ5 < circleArcAngleStart)
{
// start moved even lower
minAngle = startAngleQ5;
maxAngle = CWRUtil::Q5(ROUNDUP((int)circleArcAngleStart, circleArcIncrementQ5int));
maxAngle = MIN(maxAngle, circleArcAngleEnd); // No need to go higher than end
}
else if (startAngleQ5 < circleArcAngleEnd)
{
// start moved higher, but not higher than end
minAngle = circleArcAngleStart;
maxAngle = CWRUtil::Q5(ROUNDUP((int)startAngleQ5, circleArcIncrementQ5int));
maxAngle = MIN(maxAngle, circleArcAngleEnd); // No need to go higher than end
}
else
{
// start moved past end
minAngle = circleArcAngleStart;
maxAngle = startAngleQ5;
}
}
else
{
// start is higher than end
if (startAngleQ5 > circleArcAngleStart)
{
// start moved even higher
minAngle = CWRUtil::Q5(ROUNDDOWN((int)circleArcAngleStart, circleArcIncrementQ5int));
minAngle = MAX(minAngle, circleArcAngleEnd); // No need to go lower then end
maxAngle = startAngleQ5;
}
else if (startAngleQ5 > circleArcAngleEnd)
{
// start moved lower, but not lower than end
minAngle = CWRUtil::Q5(ROUNDDOWN((int)startAngleQ5, circleArcIncrementQ5int));
minAngle = MAX(minAngle, circleArcAngleEnd); // No need to go lower than end
maxAngle = circleArcAngleStart;
}
else
{
// start moved lower past end
minAngle = startAngleQ5;
maxAngle = circleArcAngleStart;
}
}
return getMinimalRect(minAngle, maxAngle);
}
Rect Circle::getMinimalRectForUpdatedEndAngle(const CWRUtil::Q5& endAngleQ5) const
{
CWRUtil::Q5 minAngle = CWRUtil::Q5(0); // Unused default value
CWRUtil::Q5 maxAngle = CWRUtil::Q5(0); // Unused default value
int circleArcIncrementQ5int = (int)CWRUtil::toQ5<int>(circleArcIncrement);
if (circleArcAngleStart < circleArcAngleEnd)
{
// start is smaller than end
if (endAngleQ5 > circleArcAngleEnd)
{
// end moved even higher
minAngle = CWRUtil::Q5(ROUNDDOWN((int)circleArcAngleEnd, circleArcIncrementQ5int));
minAngle = MAX(minAngle, circleArcAngleStart);
maxAngle = endAngleQ5;
}
else if (endAngleQ5 > circleArcAngleStart)
{
// end moved lower, but not past start
minAngle = CWRUtil::Q5(ROUNDDOWN((int)endAngleQ5, circleArcIncrementQ5int));
minAngle = MAX(minAngle, circleArcAngleStart); // No need to go lower than start
maxAngle = circleArcAngleEnd;
}
else
{
// end move past start
minAngle = endAngleQ5;
maxAngle = circleArcAngleEnd;
}
}
else
{
// start is higher than end
if (endAngleQ5 < circleArcAngleEnd)
{
// end moved even lower
minAngle = endAngleQ5;
maxAngle = CWRUtil::Q5(ROUNDUP((int)circleArcAngleEnd, circleArcIncrementQ5int));
maxAngle = MIN(maxAngle, circleArcAngleStart); // No need to go higher than start
}
else if (endAngleQ5 < circleArcAngleStart)
{
// end moved higher, but not higher than start
minAngle = circleArcAngleEnd;
maxAngle = CWRUtil::Q5(ROUNDUP((int)endAngleQ5, circleArcIncrementQ5int));
maxAngle = MIN(maxAngle, circleArcAngleStart);
}
else
{
// end moved past start
minAngle = circleArcAngleEnd;
maxAngle = endAngleQ5;
}
}
return getMinimalRect(minAngle, maxAngle);
}
} // namespace touchgfx

View File

@ -0,0 +1,285 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <touchgfx/widgets/canvas/Line.hpp>
namespace touchgfx
{
Line::Line()
: CanvasWidget(),
startX(0), startY(0), endX(0), endY(0),
lineWidth(CWRUtil::toQ5<int>(1)),
lineEnding(BUTT_CAP_ENDING),
lineCapArcIncrement(18)
{
Drawable::setWidthHeight(0, 0);
}
void Line::setStart(CWRUtil::Q5 xQ5, CWRUtil::Q5 yQ5)
{
if (startX == xQ5 && startY == yQ5)
{
return;
}
startX = xQ5;
startY = yQ5;
updateCachedShape();
}
void Line::updateStart(CWRUtil::Q5 xQ5, CWRUtil::Q5 yQ5)
{
if (startX == xQ5 && startY == yQ5)
{
return;
}
Rect rectBefore = getMinimalRect();
startX = xQ5;
startY = yQ5;
updateCachedShape();
Rect rectAfter = getMinimalRect();
rectBefore.expandToFit(rectAfter);
invalidateRect(rectBefore);
}
void Line::setEnd(CWRUtil::Q5 xQ5, CWRUtil::Q5 yQ5)
{
if (endX == xQ5 && endY == yQ5)
{
return;
}
endX = xQ5;
endY = yQ5;
updateCachedShape();
}
void Line::updateEnd(CWRUtil::Q5 xQ5, CWRUtil::Q5 yQ5)
{
if (endX == xQ5 && endY == yQ5)
{
return;
}
Rect rectBefore = getMinimalRect();
endX = xQ5;
endY = yQ5;
updateCachedShape();
Rect rectAfter = getMinimalRect();
rectBefore.expandToFit(rectAfter);
invalidateRect(rectBefore);
}
void Line::setLineEndingStyle(Line::LINE_ENDING_STYLE lineEndingStyle)
{
lineEnding = lineEndingStyle;
updateCachedShape();
}
Line::LINE_ENDING_STYLE Line::getLineEndingStyle() const
{
return lineEnding;
}
void Line::setCapPrecision(int precision)
{
if (precision < 1)
{
precision = 1;
}
if (precision > 180)
{
precision = 180;
}
lineCapArcIncrement = precision;
}
bool Line::drawCanvasWidget(const Rect& invalidatedArea) const
{
Canvas canvas(this, invalidatedArea);
CWRUtil::Q5 radius;
int angleInDegrees = CWRUtil::angle(xCorner[0] - startX, yCorner[0] - startY, radius);
canvas.moveTo(xCorner[0], yCorner[0]);
canvas.lineTo(xCorner[1], yCorner[1]);
if (lineEnding == ROUND_CAP_ENDING)
{
// Fixed 10 steps (steps 0 and 9 are at Corner[1] and [2])
for (int i = lineCapArcIncrement; i < 180; i += lineCapArcIncrement)
{
canvas.lineTo(endX + radius * CWRUtil::sine(angleInDegrees - i), endY - radius * CWRUtil::cosine(angleInDegrees - i));
}
}
canvas.lineTo(xCorner[2], yCorner[2]);
canvas.lineTo(xCorner[3], yCorner[3]);
if (lineEnding == ROUND_CAP_ENDING)
{
// Fixed 10 steps (steps 0 and 9 are at Corner[3] and [0])
for (int i = 180 - lineCapArcIncrement; i > 0; i -= lineCapArcIncrement)
{
canvas.lineTo(startX + radius * CWRUtil::sine(angleInDegrees + i), startY - radius * CWRUtil::cosine(angleInDegrees + i));
}
}
return canvas.render();
}
void Line::updateCachedShape()
{
CWRUtil::Q5 dx = endX - startX;
CWRUtil::Q5 dy = endY - startY;
CWRUtil::Q5 d = CWRUtil::toQ5<int>(0);
// Look for horizontal, vertical or no-line
if ((int32_t)dx == 0)
{
if ((int32_t)dy == 0)
{
xCorner[0] = xCorner[1] = xCorner[2] = xCorner[3] = startX;
yCorner[0] = yCorner[1] = yCorner[2] = yCorner[3] = startY;
return;
}
d = abs(dy);
}
else if ((int32_t)dy == 0)
{
d = abs(dx);
}
else
{
// We want to hit as close to the limit inside 32bits.
// sqrt(0x7FFFFFFF / 2) = 46340, which is roughly toQ5(1448)
static const int32_t MAXVAL = 46340;
int32_t common_divisor = gcd(abs((int32_t)dx), abs((int32_t)dy));
// First try to scale down
if (common_divisor != 1)
{
dx = CWRUtil::Q5((int32_t)dx / common_divisor);
dy = CWRUtil::Q5((int32_t)dy / common_divisor);
}
// Neither dx or dy is zero, find the largest multiplier / smallest divisor to stay within limit
if (abs((int32_t)dx) <= MAXVAL || abs((int32_t)dy) <= MAXVAL)
{
// Look for largest multiplier
int32_t mulx = MAXVAL / abs((int32_t)dx);
int32_t muly = MAXVAL / abs((int32_t)dy);
int32_t mult = MIN(mulx, muly);
dx = CWRUtil::Q5(mult * (int32_t)dx);
dy = CWRUtil::Q5(mult * (int32_t)dy);
}
else
{
// Look for smallest divisor
int32_t divx = abs((int32_t)dx) / MAXVAL;
int32_t divy = abs((int32_t)dy) / MAXVAL;
int32_t divi = MAX(divx, divy) + 1;
dx = CWRUtil::Q5((int32_t)dx / divi);
dy = CWRUtil::Q5((int32_t)dy / divi);
}
d = CWRUtil::sqrtQ10(dy * dy + dx * dx);
}
dy = CWRUtil::muldivQ5(lineWidth, dy, d) / 2;
dx = CWRUtil::muldivQ5(lineWidth, dx, d) / 2;
switch (lineEnding)
{
case BUTT_CAP_ENDING:
xCorner[0] = startX - dy;
yCorner[0] = startY + dx;
xCorner[1] = endX - dy;
yCorner[1] = endY + dx;
xCorner[2] = endX + dy;
yCorner[2] = endY - dx;
xCorner[3] = startX + dy;
yCorner[3] = startY - dx;
break;
case ROUND_CAP_ENDING:
// Nothing cached, calculated on each draw, but extremes are same as SQUARE_CAP_ENDING, so
// Fall Through (for calculations)
default:
case SQUARE_CAP_ENDING:
xCorner[0] = startX - dy - dx;
yCorner[0] = startY + dx - dy;
xCorner[1] = endX - dy + dx;
yCorner[1] = endY + dx + dy;
xCorner[2] = endX + dy + dx;
yCorner[2] = endY - dx + dy;
xCorner[3] = startX + dy - dx;
yCorner[3] = startY - dx - dy;
break;
}
CWRUtil::Q5 xMin = xCorner[0];
CWRUtil::Q5 xMax = xCorner[0];
CWRUtil::Q5 yMin = yCorner[0];
CWRUtil::Q5 yMax = yCorner[0];
for (int i = 1; i < 4; i++)
{
if (xCorner[i] < xMin)
{
xMin = xCorner[i];
}
if (xCorner[i] > xMax)
{
xMax = xCorner[i];
}
if (yCorner[i] < yMin)
{
yMin = yCorner[i];
}
if (yCorner[i] > yMax)
{
yMax = yCorner[i];
}
}
int16_t minX = xMin.to<int>();
int16_t minY = yMin.to<int>();
int16_t maxX = xMax.to<int>();
int16_t maxY = yMax.to<int>();
minimalRect = Rect(minX, minY, maxX - minX + 1, maxY - minY + 1);
if (lineEnding == ROUND_CAP_ENDING)
{
xCorner[0] = startX - dy;
yCorner[0] = startY + dx;
xCorner[1] = endX - dy;
yCorner[1] = endY + dx;
xCorner[2] = endX + dy;
yCorner[2] = endY - dx;
xCorner[3] = startX + dy;
yCorner[3] = startY - dx;
}
}
Rect Line::getMinimalRect() const
{
return minimalRect;
}
void Line::updateLengthAndAngle(CWRUtil::Q5 length, CWRUtil::Q5 angle)
{
updateEnd(startX + length * CWRUtil::sine(angle), startY - length * CWRUtil::cosine(angle));
}
} // namespace touchgfx

View File

@ -0,0 +1,68 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <platform/driver/lcd/LCD8bpp_ABGR2222.hpp>
#include <touchgfx/Color.hpp>
#include <touchgfx/widgets/canvas/PainterABGR2222.hpp>
namespace touchgfx
{
void PainterABGR2222::render(uint8_t* ptr, int x, int xAdjust, int /*y*/, unsigned count, const uint8_t* covers)
{
uint8_t* p = ptr + (x + xAdjust);
const uint8_t totalAlpha = LCD::div255(widgetAlpha * painterAlpha);
if (totalAlpha == 0xFF)
{
do
{
const uint8_t alpha = *covers++;
if (alpha == 0xFF)
{
*p = painterColor;
}
else
{
*p = mixColors(painterRed, painterGreen, painterBlue, *p, alpha);
}
p++;
} while (--count != 0);
}
else
{
do
{
const uint8_t alpha = LCD::div255((*covers++) * totalAlpha);
if (alpha == 0xFF)
{
*p = painterColor;
}
else
{
*p = mixColors(painterRed, painterGreen, painterBlue, *p, alpha);
}
p++;
} while (--count != 0);
}
}
bool PainterABGR2222::renderNext(uint8_t& red, uint8_t& green, uint8_t& blue, uint8_t& alpha)
{
red = painterRed;
green = painterGreen;
blue = painterBlue;
alpha = painterAlpha;
return true;
}
} // namespace touchgfx

View File

@ -0,0 +1,139 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <touchgfx/widgets/canvas/PainterABGR2222Bitmap.hpp>
namespace touchgfx
{
void PainterABGR2222Bitmap::setBitmap(const Bitmap& bmp)
{
bitmap = bmp;
assert((bitmap.getId() == BITMAP_INVALID || bitmap.getFormat() == Bitmap::ABGR2222) && "The chosen painter only works with ABGR2222 bitmaps");
bitmapRectToFrameBuffer = bitmap.getRect();
DisplayTransformation::transformDisplayToFrameBuffer(bitmapRectToFrameBuffer);
}
void PainterABGR2222Bitmap::render(uint8_t* ptr,
int x,
int xAdjust,
int y,
unsigned count,
const uint8_t* covers)
{
uint8_t* p = ptr + (x + xAdjust);
currentX = x + areaOffsetX;
currentY = y + areaOffsetY;
if (!renderInit())
{
return;
}
if (currentX + (int)count > bitmapRectToFrameBuffer.width)
{
count = bitmapRectToFrameBuffer.width - currentX;
}
const uint8_t totalAlpha = LCD::div255(widgetAlpha * painterAlpha);
const uint8_t* src = bitmapABGR2222Pointer;
if (totalAlpha == 0xFF)
{
do
{
const uint8_t srcAlpha = ((*src) >> 6) * 0x55;
const uint8_t alpha = LCD::div255((*covers++) * srcAlpha);
if (alpha == 0xFF)
{
// Solid pixel
*p = *src;
}
else if (alpha)
{
// Non-Transparent pixel
*p = mixColors(*src, *p, alpha);
}
p++;
src++;
} while (--count != 0);
}
else
{
do
{
const uint8_t srcAlpha = ((*src) >> 6) * 0x55;
const uint8_t alpha = LCD::div255((*covers++) * srcAlpha);
if (alpha) // This can never get to max=0xFF*0xFF as totalAlpha<255
{
// Non-Transparent pixel
*p = mixColors(*src, *p, alpha);
}
p++;
src++;
} while (--count != 0);
}
}
bool PainterABGR2222Bitmap::renderInit()
{
bitmapABGR2222Pointer = 0;
if (bitmap.getId() == BITMAP_INVALID)
{
return false;
}
if ((currentX >= bitmapRectToFrameBuffer.width) || (currentY >= bitmapRectToFrameBuffer.height))
{
// Outside bitmap area, do not draw anything
// Consider the following instead of "return" to get a tiled image:
// currentX %= bitmapRectToFrameBuffer.width
// currentY %= bitmapRectToFrameBuffer.height
return false;
}
if (bitmap.getFormat() == Bitmap::ABGR2222)
{
bitmapABGR2222Pointer = bitmap.getData();
if (!bitmapABGR2222Pointer)
{
return false;
}
bitmapABGR2222Pointer += currentX + currentY * bitmapRectToFrameBuffer.width;
return true;
}
return false;
}
bool PainterABGR2222Bitmap::renderNext(uint8_t& red, uint8_t& green, uint8_t& blue, uint8_t& alpha)
{
if (currentX >= bitmapRectToFrameBuffer.width)
{
return false;
}
else if (bitmapABGR2222Pointer != 0)
{
uint16_t abgr2222 = *bitmapABGR2222Pointer++;
red = LCD8bpp_ABGR2222::getRedFromColor(abgr2222);
green = LCD8bpp_ABGR2222::getGreenFromColor(abgr2222);
blue = LCD8bpp_ABGR2222::getBlueFromColor(abgr2222);
alpha = (abgr2222 >> 6) * 0x55; // To get full range 0-0xFF
}
// Apply given alpha from setAlpha()
alpha = LCD::div255(alpha * painterAlpha);
return true;
}
} // namespace touchgfx

View File

@ -0,0 +1,68 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <platform/driver/lcd/LCD8bpp_ARGB2222.hpp>
#include <touchgfx/Color.hpp>
#include <touchgfx/widgets/canvas/PainterARGB2222.hpp>
namespace touchgfx
{
void PainterARGB2222::render(uint8_t* ptr, int x, int xAdjust, int /*y*/, unsigned count, const uint8_t* covers)
{
uint8_t* p = ptr + (x + xAdjust);
const uint8_t totalAlpha = LCD::div255(widgetAlpha * painterAlpha);
if (totalAlpha == 0xFF)
{
do
{
const uint8_t alpha = *covers++;
if (alpha == 0xFF)
{
*p = painterColor;
}
else
{
*p = mixColors(painterRed, painterGreen, painterBlue, *p, alpha);
}
p++;
} while (--count != 0);
}
else
{
do
{
const uint8_t alpha = LCD::div255((*covers++) * totalAlpha);
if (alpha == 0xFF)
{
*p = painterColor;
}
else
{
*p = mixColors(painterRed, painterGreen, painterBlue, *p, alpha);
}
p++;
} while (--count != 0);
}
}
bool PainterARGB2222::renderNext(uint8_t& red, uint8_t& green, uint8_t& blue, uint8_t& alpha)
{
red = painterRed;
green = painterGreen;
blue = painterBlue;
alpha = painterAlpha;
return true;
}
} // namespace touchgfx

View File

@ -0,0 +1,139 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <touchgfx/widgets/canvas/PainterARGB2222Bitmap.hpp>
namespace touchgfx
{
void PainterARGB2222Bitmap::setBitmap(const Bitmap& bmp)
{
bitmap = bmp;
assert((bitmap.getId() == BITMAP_INVALID || bitmap.getFormat() == Bitmap::ARGB2222) && "The chosen painter only works with ARGB2222 bitmaps");
bitmapRectToFrameBuffer = bitmap.getRect();
DisplayTransformation::transformDisplayToFrameBuffer(bitmapRectToFrameBuffer);
}
void PainterARGB2222Bitmap::render(uint8_t* ptr,
int x,
int xAdjust,
int y,
unsigned count,
const uint8_t* covers)
{
uint8_t* p = ptr + (x + xAdjust);
currentX = x + areaOffsetX;
currentY = y + areaOffsetY;
if (!renderInit())
{
return;
}
if (currentX + (int)count > bitmapRectToFrameBuffer.width)
{
count = bitmapRectToFrameBuffer.width - currentX;
}
const uint8_t totalAlpha = LCD::div255(widgetAlpha * painterAlpha);
const uint8_t* src = bitmapARGB2222Pointer;
if (totalAlpha == 0xFF)
{
do
{
const uint8_t srcAlpha = ((*src) >> 6) * 0x55;
const uint8_t alpha = LCD::div255((*covers++) * srcAlpha);
if (alpha == 0xFF)
{
// Solid pixel
*p = *src;
}
else if (alpha)
{
// Non-Transparent pixel
*p = mixColors(*src, *p, alpha);
}
p++;
src++;
} while (--count != 0);
}
else
{
do
{
const uint8_t srcAlpha = ((*src) >> 6) * 0x55;
const uint8_t alpha = LCD::div255((*covers++) * srcAlpha);
if (alpha) // This can never get to max=0xFF*0xFF as totalAlpha<255
{
// Non-Transparent pixel
*p = mixColors(*src, *p, alpha);
}
p++;
src++;
} while (--count != 0);
}
}
bool PainterARGB2222Bitmap::renderInit()
{
bitmapARGB2222Pointer = 0;
if (bitmap.getId() == BITMAP_INVALID)
{
return false;
}
if ((currentX >= bitmapRectToFrameBuffer.width) || (currentY >= bitmapRectToFrameBuffer.height))
{
// Outside bitmap area, do not draw anything
// Consider the following instead of "return" to get a tiled image:
// currentX %= bitmapRectToFrameBuffer.width
// currentY %= bitmapRectToFrameBuffer.height
return false;
}
if (bitmap.getFormat() == Bitmap::ARGB2222)
{
bitmapARGB2222Pointer = bitmap.getData();
if (!bitmapARGB2222Pointer)
{
return false;
}
bitmapARGB2222Pointer += currentX + currentY * bitmapRectToFrameBuffer.width;
return true;
}
return false;
}
bool PainterARGB2222Bitmap::renderNext(uint8_t& red, uint8_t& green, uint8_t& blue, uint8_t& alpha)
{
if (currentX >= bitmapRectToFrameBuffer.width)
{
return false;
}
else if (bitmapARGB2222Pointer != 0)
{
uint16_t ARGB2222 = *bitmapARGB2222Pointer++;
red = LCD8bpp_ARGB2222::getRedFromColor(ARGB2222);
green = LCD8bpp_ARGB2222::getGreenFromColor(ARGB2222);
blue = LCD8bpp_ARGB2222::getBlueFromColor(ARGB2222);
alpha = (ARGB2222 >> 6) * 0x55; // To get full range 0-0xFF
}
// Apply given alpha from setAlpha()
alpha = LCD::div255(alpha * painterAlpha);
return true;
}
} // namespace touchgfx

View File

@ -0,0 +1,77 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <touchgfx/widgets/canvas/PainterARGB8888.hpp>
namespace touchgfx
{
void PainterARGB8888::render(uint8_t* ptr, int x, int xAdjust, int /*y*/, unsigned count, const uint8_t* covers)
{
uint8_t* p = reinterpret_cast<uint8_t*>(ptr) + ((x + xAdjust) * 4);
uint8_t pByte;
const uint8_t totalAlpha = LCD::div255(widgetAlpha * painterAlpha);
if (totalAlpha == 0xFF)
{
do
{
const uint8_t alpha = *covers++;
if (alpha == 0xFF)
{
*p++ = painterBlue;
*p++ = painterGreen;
*p++ = painterRed;
*p++ = 0xff;
}
else
{
const uint8_t ialpha = 0xFF - alpha;
pByte = *p;
*p++ = LCD::div255(painterBlue * alpha + pByte * ialpha);
pByte = *p;
*p++ = LCD::div255(painterGreen * alpha + pByte * ialpha);
pByte = *p;
*p++ = LCD::div255(painterRed * alpha + pByte * ialpha);
pByte = *p;
*p++ = pByte + alpha - LCD::div255(pByte * alpha);
}
} while (--count != 0);
}
else if (totalAlpha != 0)
{
do
{
const uint8_t alpha = LCD::div255(*covers++ * totalAlpha);
const uint8_t ialpha = 0xFF - alpha;
pByte = *p;
*p++ = LCD::div255(painterBlue * alpha + pByte * ialpha);
pByte = *p;
*p++ = LCD::div255(painterGreen * alpha + pByte * ialpha);
pByte = *p;
*p++ = LCD::div255(painterRed * alpha + pByte * ialpha);
pByte = *p;
*p++ = pByte + alpha - LCD::div255(pByte * alpha);
} while (--count != 0);
}
}
bool PainterARGB8888::renderNext(uint8_t& red, uint8_t& green, uint8_t& blue, uint8_t& alpha)
{
red = painterRed;
green = painterGreen;
blue = painterBlue;
alpha = painterAlpha;
return true;
}
} // namespace touchgfx

View File

@ -0,0 +1,340 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <touchgfx/widgets/canvas/PainterARGB8888Bitmap.hpp>
namespace touchgfx
{
void PainterARGB8888Bitmap::setBitmap(const Bitmap& bmp)
{
bitmap = bmp;
assert((bitmap.getId() == BITMAP_INVALID || bitmap.getFormat() == Bitmap::RGB565 || bitmap.getFormat() == Bitmap::RGB888 || bitmap.getFormat() == Bitmap::ARGB8888) && "The chosen painter only works with RGB565, RGB888 and ARGB8888 bitmaps");
bitmapRectToFrameBuffer = bitmap.getRect();
DisplayTransformation::transformDisplayToFrameBuffer(bitmapRectToFrameBuffer);
}
void PainterARGB8888Bitmap::render(uint8_t* ptr, int x, int xAdjust, int y, unsigned count, const uint8_t* covers)
{
uint8_t* RESTRICT p = ptr + ((x + xAdjust) * 4);
currentX = x + areaOffsetX;
currentY = y + areaOffsetY;
if (!renderInit())
{
return;
}
if (currentX + (int)count > bitmapRectToFrameBuffer.width)
{
count = bitmapRectToFrameBuffer.width - currentX;
}
const uint8_t totalAlpha = LCD::div255(widgetAlpha * painterAlpha);
if (bitmap.getFormat() == Bitmap::ARGB8888)
{
const uint32_t* RESTRICT src = bitmapARGB8888Pointer;
if (totalAlpha == 0xFF)
{
do
{
const uint8_t srcAlpha = (*src) >> 24;
const uint8_t alpha = LCD::div255((*covers++) * srcAlpha);
if (alpha == 0xFF)
{
// Solid pixel
*p++ = (*src); // Blue
*p++ = (*src) >> 8; // Green
*p++ = (*src) >> 16; // Red
*p++ = 0xff; // Alpha
}
else
{
// Non-Transparent pixel
const uint8_t ialpha = 0xFF - alpha;
uint8_t pByte = *p;
uint8_t cByte = (*src);
*p++ = LCD::div255(cByte * alpha + pByte * ialpha);
pByte = *p;
cByte = (*src) >> 8;
*p++ = LCD::div255(cByte * alpha + pByte * ialpha);
pByte = *p;
cByte = (*src) >> 16;
*p++ = LCD::div255(cByte * alpha + pByte * ialpha);
pByte = *p;
cByte = (*src) >> 24;
*p++ = pByte + alpha - LCD::div255(pByte * alpha);
}
src++;
} while (--count != 0);
}
else
{
do
{
const uint8_t srcAlpha = (*src) >> 24;
const uint8_t alpha = LCD::div255((*covers++) * LCD::div255(srcAlpha * totalAlpha));
if (alpha)
{
const uint8_t ialpha = 0xFF - alpha;
uint8_t pByte = *p;
uint8_t cByte = (*src);
*p++ = LCD::div255(cByte * alpha + pByte * ialpha);
pByte = *p;
cByte = (*src) >> 8;
*p++ = LCD::div255(cByte * alpha + pByte * ialpha);
pByte = *p;
cByte = (*src) >> 16;
*p++ = LCD::div255(cByte * alpha + pByte * ialpha);
pByte = *p;
cByte = (*src) >> 24;
*p++ = pByte + alpha - LCD::div255(pByte * alpha);
}
else
{
p += 4;
}
src++;
} while (--count != 0);
}
}
else if (bitmap.getFormat() == Bitmap::RGB888)
{
const uint8_t* RESTRICT src = bitmapRGB888Pointer;
if (totalAlpha == 0xFF)
{
do
{
const uint8_t alpha = *covers++;
if (alpha == 0xFF)
{
// Solid pixel
*p++ = *src++; // Blue
*p++ = *src++; // Green
*p++ = *src++; // Red
*p++ = 0xff; // Alpha
}
else
{
// Non-Transparent pixel
const uint8_t ialpha = 0xFF - alpha;
uint8_t pByte = *p;
uint8_t cByte = *src++;
*p++ = LCD::div255(cByte * alpha + pByte * ialpha);
pByte = *p;
cByte = *src++;
*p++ = LCD::div255(cByte * alpha + pByte * ialpha);
pByte = *p;
cByte = *src++;
*p++ = LCD::div255(cByte * alpha + pByte * ialpha);
pByte = *p;
*p++ = pByte + alpha - LCD::div255(pByte * alpha);
}
} while (--count != 0);
}
else
{
do
{
const uint8_t alpha = LCD::div255((*covers++) * totalAlpha);
if (alpha)
{
const uint8_t ialpha = 0xFF - alpha;
uint8_t pByte = *p;
uint8_t cByte = *src++;
*p++ = LCD::div255(cByte * alpha + pByte * ialpha);
pByte = *p;
cByte = *src++;
*p++ = LCD::div255(cByte * alpha + pByte * ialpha);
pByte = *p;
cByte = *src++;
*p++ = LCD::div255(cByte * alpha + pByte * ialpha);
pByte = *p;
*p++ = pByte + alpha - LCD::div255(pByte * alpha);
}
else
{
p += 4;
src += 3;
}
} while (--count != 0);
}
}
else if (bitmap.getFormat() == Bitmap::RGB565)
{
const uint16_t* RESTRICT src = bitmapRGB565Pointer;
if (totalAlpha == 0xFF)
{
do
{
const uint8_t alpha = *covers++;
const uint16_t srcpix = *src++;
uint16_t red = (srcpix & 0xF800) >> 11;
uint16_t green = (srcpix & 0x07E0) >> 5;
uint16_t blue = srcpix & 0x001F;
red = (red * 527 + 23) >> 6;
green = (green * 259 + 33) >> 6;
blue = (blue * 527 + 23) >> 6;
if (alpha == 0xFF)
{
// Solid pixel
*p++ = (uint8_t)blue;
*p++ = (uint8_t)green;
*p++ = (uint8_t)red;
*p++ = 0xff;
}
else
{
// Non-Transparent pixel
const uint8_t ialpha = 0xFF - alpha;
uint8_t pByte = *p;
*p++ = LCD::div255(blue * alpha + pByte * ialpha);
pByte = *p;
*p++ = LCD::div255(green * alpha + pByte * ialpha);
pByte = *p;
*p++ = LCD::div255(red * alpha + pByte * ialpha);
pByte = *p;
*p++ = pByte + alpha - LCD::div255(pByte * alpha);
}
} while (--count != 0);
}
else
{
do
{
const uint8_t alpha = LCD::div255((*covers++) * totalAlpha);
uint16_t srcpix = *src++;
uint16_t red = (srcpix & 0xF800) >> 11;
uint16_t green = (srcpix & 0x07E0) >> 5;
uint16_t blue = srcpix & 0x001F;
red = (red * 527 + 23) >> 6;
green = (green * 259 + 33) >> 6;
blue = (blue * 527 + 23) >> 6;
if (alpha)
{
const uint8_t ialpha = 0xFF - alpha;
uint8_t pByte = *p;
*p++ = LCD::div255(blue * alpha + pByte * ialpha);
pByte = *p;
*p++ = LCD::div255(green * alpha + pByte * ialpha);
pByte = *p;
*p++ = LCD::div255(red * alpha + pByte * ialpha);
pByte = *p;
*p++ = pByte + alpha - LCD::div255(pByte * alpha);
}
else
{
p += 4;
}
} while (--count != 0);
}
}
}
bool PainterARGB8888Bitmap::renderInit()
{
bitmapARGB8888Pointer = 0;
bitmapRGB565Pointer = 0;
bitmapRGB888Pointer = 0;
if (bitmap.getId() == BITMAP_INVALID)
{
return false;
}
if ((currentX >= bitmapRectToFrameBuffer.width) || (currentY >= bitmapRectToFrameBuffer.height))
{
// Outside bitmap area, do not draw anything
// Consider the following instead of "return" to get a tiled image:
// currentX %= bitmapRectToFrameBuffer.width
// currentY %= bitmapRectToFrameBuffer.height
return false;
}
if (bitmap.getFormat() == Bitmap::ARGB8888)
{
bitmapARGB8888Pointer = (const uint32_t*)bitmap.getData();
if (!bitmapARGB8888Pointer)
{
return false;
}
bitmapARGB8888Pointer += currentX + currentY * bitmapRectToFrameBuffer.width;
return true;
}
if (bitmap.getFormat() == Bitmap::RGB888)
{
bitmapRGB888Pointer = (const uint8_t*)bitmap.getData();
if (!bitmapRGB888Pointer)
{
return false;
}
bitmapRGB888Pointer += (currentX + currentY * bitmapRectToFrameBuffer.width) * 3;
return true;
}
if (bitmap.getFormat() == Bitmap::RGB565)
{
bitmapRGB565Pointer = (const uint16_t*)bitmap.getData();
if (!bitmapRGB565Pointer)
{
return false;
}
bitmapRGB565Pointer += currentX + currentY * bitmapRectToFrameBuffer.width;
return true;
}
return false;
}
bool PainterARGB8888Bitmap::renderNext(uint8_t& red, uint8_t& green, uint8_t& blue, uint8_t& alpha)
{
if (currentX >= bitmapRectToFrameBuffer.width)
{
return false;
}
if (bitmapARGB8888Pointer != 0)
{
uint32_t argb8888 = *bitmapARGB8888Pointer++;
alpha = (argb8888 >> 24) & 0xFF;
red = (argb8888 >> 16) & 0xFF;
green = (argb8888 >> 8) & 0xFF;
blue = (argb8888) & 0xFF;
}
else if (bitmapRGB888Pointer != 0)
{
blue = *bitmapRGB888Pointer++;
green = *bitmapRGB888Pointer++;
red = *bitmapRGB888Pointer++;
alpha = 0xff;
}
else if (bitmapRGB565Pointer != 0)
{
uint16_t srcpix = *bitmapRGB565Pointer++;
red = (srcpix & 0xF800) >> 11;
green = (srcpix & 0x07E0) >> 5;
blue = srcpix & 0x001F;
red = (red * 527 + 23) >> 6;
green = (green * 259 + 33) >> 6;
blue = (blue * 527 + 23) >> 6;
alpha = 0xff;
}
// Apply given alpha from setAlpha()
alpha = LCD::div255(alpha * painterAlpha);
return true;
}
} // namespace touchgfx

View File

@ -0,0 +1,311 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <touchgfx/widgets/canvas/PainterARGB8888L8Bitmap.hpp>
namespace touchgfx
{
void PainterARGB8888L8Bitmap::setBitmap(const Bitmap& bmp)
{
bitmap = bmp;
assert((bitmap.getId() == BITMAP_INVALID || bitmap.getFormat() == Bitmap::L8) && "The chosen painter only works with L8 bitmaps");
bitmapRectToFrameBuffer = bitmap.getRect();
DisplayTransformation::transformDisplayToFrameBuffer(bitmapRectToFrameBuffer);
}
void PainterARGB8888L8Bitmap::render(uint8_t* ptr, int x, int xAdjust, int y, unsigned count, const uint8_t* covers)
{
uint8_t* RESTRICT p = ptr + ((x + xAdjust) * 4);
currentX = x + areaOffsetX;
currentY = y + areaOffsetY;
if (!renderInit())
{
return;
}
if (currentX + (int)count > bitmapRectToFrameBuffer.width)
{
count = bitmapRectToFrameBuffer.width - currentX;
}
const uint8_t totalAlpha = LCD::div255(widgetAlpha * painterAlpha);
switch ((Bitmap::ClutFormat)((const uint16_t*)bitmapExtraPointer)[-2])
{
case Bitmap::CLUT_FORMAT_L8_RGB565:
if (totalAlpha == 0xFF)
{
do
{
const uint16_t srcpix = ((const uint16_t*)bitmapExtraPointer)[*bitmapPointer++];
const uint8_t alpha = *covers++;
uint16_t red = (srcpix & 0xF800) >> 11;
uint16_t green = (srcpix & 0x07E0) >> 5;
uint16_t blue = srcpix & 0x001F;
red = (red * 527 + 23) >> 6;
green = (green * 259 + 33) >> 6;
blue = (blue * 527 + 23) >> 6;
if (alpha == 0xFF)
{
// Solid pixel
*p++ = (uint8_t)blue;
*p++ = (uint8_t)green;
*p++ = (uint8_t)red;
*p++ = 0xff;
}
else
{
// Non-Transparent pixel
const uint8_t ialpha = 0xFF - alpha;
uint8_t pByte = *p;
*p++ = LCD::div255(blue * alpha + pByte * ialpha);
pByte = *p;
*p++ = LCD::div255(green * alpha + pByte * ialpha);
pByte = *p;
*p++ = LCD::div255(red * alpha + pByte * ialpha);
pByte = *p;
*p++ = pByte + alpha - LCD::div255(pByte * alpha);
}
} while (--count != 0);
}
else
{
do
{
const uint16_t srcpix = ((const uint16_t*)bitmapExtraPointer)[*bitmapPointer++];
const uint8_t alpha = LCD::div255((*covers++) * totalAlpha);
uint16_t red = (srcpix & 0xF800) >> 11;
uint16_t green = (srcpix & 0x07E0) >> 5;
uint16_t blue = srcpix & 0x001F;
red = (red * 527 + 23) >> 6;
green = (green * 259 + 33) >> 6;
blue = (blue * 527 + 23) >> 6;
if (alpha)
{
const uint8_t ialpha = 0xFF - alpha;
uint8_t pByte = *p;
*p++ = LCD::div255(blue * alpha + pByte * ialpha);
pByte = *p;
*p++ = LCD::div255(green * alpha + pByte * ialpha);
pByte = *p;
*p++ = LCD::div255(red * alpha + pByte * ialpha);
pByte = *p;
*p++ = pByte + alpha - LCD::div255(pByte * alpha);
}
else
{
p += 4;
}
} while (--count != 0);
}
break;
case Bitmap::CLUT_FORMAT_L8_RGB888:
if (totalAlpha == 0xFF)
{
do
{
const uint8_t* src = bitmapExtraPointer + *bitmapPointer++ * 3;
const uint8_t alpha = *covers++;
if (alpha == 0xFF)
{
// Solid pixel
*p++ = *src++; // Blue
*p++ = *src++; // Green
*p++ = *src; // Red
*p++ = 0xff; // Alpha
}
else
{
// Non-Transparent pixel
const uint8_t ialpha = 0xFF - alpha;
uint8_t pByte = *p;
*p++ = LCD::div255(*src++ * alpha + pByte * ialpha);
pByte = *p;
*p++ = LCD::div255(*src++ * alpha + pByte * ialpha);
pByte = *p;
*p++ = LCD::div255(*src * alpha + pByte * ialpha);
pByte = *p;
*p++ = pByte + alpha - LCD::div255(pByte * alpha);
}
} while (--count != 0);
}
else
{
do
{
const uint8_t alpha = LCD::div255((*covers++) * totalAlpha);
if (alpha)
{
const uint8_t* src = bitmapExtraPointer + *bitmapPointer++ * 3;
const uint8_t ialpha = 0xFF - alpha;
uint8_t pByte = *p;
*p++ = LCD::div255(*src++ * alpha + pByte * ialpha);
pByte = *p;
*p++ = LCD::div255(*src++ * alpha + pByte * ialpha);
pByte = *p;
*p++ = LCD::div255(*src * alpha + pByte * ialpha);
pByte = *p;
*p++ = pByte + alpha - LCD::div255(pByte * alpha);
}
else
{
p += 4;
}
} while (--count != 0);
}
break;
case Bitmap::CLUT_FORMAT_L8_ARGB8888:
if (totalAlpha == 0xFF)
{
do
{
uint32_t src = ((const uint32_t*)bitmapExtraPointer)[*bitmapPointer++];
const uint8_t srcAlpha = src >> 24;
const uint8_t alpha = LCD::div255((*covers++) * srcAlpha);
if (alpha == 0xFF)
{
// Solid pixel
*p++ = src; // Blue
*p++ = src >> 8; // Green
*p++ = src >> 16; // Red
*p++ = 0xff; // Alpha
}
else
{
// Non-Transparent pixel
const uint8_t ialpha = 0xFF - alpha;
uint8_t pByte = *p;
*p++ = LCD::div255((src & 0xFF) * alpha + pByte * ialpha);
pByte = *p;
*p++ = LCD::div255(((src >> 8) & 0xFF) * alpha + pByte * ialpha);
pByte = *p;
*p++ = LCD::div255(((src >> 16) & 0xFF) * alpha + pByte * ialpha);
pByte = *p;
*p++ = pByte + alpha - LCD::div255(pByte * alpha);
}
} while (--count != 0);
}
else
{
do
{
uint32_t src = ((const uint32_t*)bitmapExtraPointer)[*bitmapPointer++];
const uint8_t srcAlpha = src >> 24;
const uint8_t alpha = LCD::div255((*covers++) * LCD::div255(srcAlpha * totalAlpha));
if (alpha)
{
const uint8_t ialpha = 0xFF - alpha;
uint8_t pByte = *p;
*p++ = LCD::div255((src & 0xFF) * alpha + pByte * ialpha);
pByte = *p;
*p++ = LCD::div255(((src >> 8) & 0xFF) * alpha + pByte * ialpha);
pByte = *p;
*p++ = LCD::div255(((src >> 16) & 0xFF) * alpha + pByte * ialpha);
pByte = *p;
*p++ = pByte + alpha - LCD::div255(pByte * alpha);
}
else
{
p += 4;
}
} while (--count != 0);
}
break;
}
}
bool PainterARGB8888L8Bitmap::renderInit()
{
bitmapPointer = 0;
bitmapExtraPointer = 0;
if (bitmap.getId() == BITMAP_INVALID)
{
return false;
}
if ((currentX >= bitmapRectToFrameBuffer.width) || (currentY >= bitmapRectToFrameBuffer.height))
{
// Outside bitmap area, do not draw anything
// Consider the following instead of "return" to get a tiled image:
// currentX %= bitmapRectToFrameBuffer.width
// currentY %= bitmapRectToFrameBuffer.height
return false;
}
if (bitmap.getFormat() == Bitmap::L8)
{
bitmapPointer = bitmap.getData();
if (!bitmapPointer)
{
return false;
}
bitmapPointer += currentX + currentY * bitmapRectToFrameBuffer.width;
bitmapExtraPointer = bitmap.getExtraData();
assert((bitmapExtraPointer != 0 && (*(const uint16_t*)bitmapExtraPointer == Bitmap::CLUT_FORMAT_L8_RGB565 || *(const uint16_t*)bitmapExtraPointer == Bitmap::CLUT_FORMAT_L8_RGB888 || *(const uint16_t*)bitmapExtraPointer == Bitmap::CLUT_FORMAT_L8_ARGB8888)));
bitmapExtraPointer += 4; // Skip header
return true;
}
return false;
}
bool PainterARGB8888L8Bitmap::renderNext(uint8_t& red, uint8_t& green, uint8_t& blue, uint8_t& alpha)
{
if (currentX >= bitmapRectToFrameBuffer.width)
{
return false;
}
switch ((Bitmap::ClutFormat)((const uint16_t*)bitmapExtraPointer)[-2])
{
case Bitmap::CLUT_FORMAT_L8_RGB565:
{
uint16_t srcpix = ((const uint16_t*)bitmapExtraPointer)[*bitmapPointer++];
red = (srcpix & 0xF800) >> 11;
green = (srcpix & 0x07E0) >> 5;
blue = srcpix & 0x001F;
red = (red * 527 + 23) >> 6;
green = (green * 259 + 33) >> 6;
blue = (blue * 527 + 23) >> 6;
alpha = 0xff;
}
break;
case Bitmap::CLUT_FORMAT_L8_RGB888:
{
const uint8_t* clut = bitmapExtraPointer + *bitmapPointer++ * 3;
blue = *clut++;
green = *clut++;
red = *clut;
alpha = 0xff;
}
break;
case Bitmap::CLUT_FORMAT_L8_ARGB8888:
{
uint32_t argb8888 = ((const uint32_t*)bitmapExtraPointer)[*bitmapPointer++];
alpha = (argb8888 >> 24) & 0xFF;
red = (argb8888 >> 16) & 0xFF;
green = (argb8888 >> 8) & 0xFF;
blue = (argb8888) & 0xFF;
}
break;
}
// Apply given alpha from setAlpha()
alpha = LCD::div255(alpha * painterAlpha);
return true;
}
} // namespace touchgfx

View File

@ -0,0 +1,68 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <platform/driver/lcd/LCD8bpp_BGRA2222.hpp>
#include <touchgfx/Color.hpp>
#include <touchgfx/widgets/canvas/PainterBGRA2222.hpp>
namespace touchgfx
{
void PainterBGRA2222::render(uint8_t* ptr, int x, int xAdjust, int /*y*/, unsigned count, const uint8_t* covers)
{
uint8_t* p = ptr + (x + xAdjust);
const uint8_t totalAlpha = LCD::div255(widgetAlpha * painterAlpha);
if (totalAlpha == 0xFF)
{
do
{
const uint8_t alpha = *covers++;
if (alpha == 0xFF)
{
*p = painterColor;
}
else
{
*p = mixColors(painterRed, painterGreen, painterBlue, *p, alpha);
}
p++;
} while (--count != 0);
}
else
{
do
{
const uint8_t alpha = LCD::div255((*covers++) * totalAlpha);
if (alpha == 0xFF)
{
*p = painterColor;
}
else
{
*p = mixColors(painterRed, painterGreen, painterBlue, *p, alpha);
}
p++;
} while (--count != 0);
}
}
bool PainterBGRA2222::renderNext(uint8_t& red, uint8_t& green, uint8_t& blue, uint8_t& alpha)
{
red = painterRed;
green = painterGreen;
blue = painterBlue;
alpha = painterAlpha;
return true;
}
} // namespace touchgfx

View File

@ -0,0 +1,139 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <touchgfx/widgets/canvas/PainterBGRA2222Bitmap.hpp>
namespace touchgfx
{
void PainterBGRA2222Bitmap::setBitmap(const Bitmap& bmp)
{
bitmap = bmp;
assert((bitmap.getId() == BITMAP_INVALID || bitmap.getFormat() == Bitmap::BGRA2222) && "The chosen painter only works with BGRA2222 bitmaps");
bitmapRectToFrameBuffer = bitmap.getRect();
DisplayTransformation::transformDisplayToFrameBuffer(bitmapRectToFrameBuffer);
}
void PainterBGRA2222Bitmap::render(uint8_t* ptr,
int x,
int xAdjust,
int y,
unsigned count,
const uint8_t* covers)
{
uint8_t* p = ptr + (x + xAdjust);
currentX = x + areaOffsetX;
currentY = y + areaOffsetY;
if (!renderInit())
{
return;
}
if (currentX + (int)count > bitmapRectToFrameBuffer.width)
{
count = bitmapRectToFrameBuffer.width - currentX;
}
const uint8_t totalAlpha = LCD::div255(widgetAlpha * painterAlpha);
const uint8_t* src = bitmapBGRA2222Pointer;
if (totalAlpha == 0xFF)
{
do
{
const uint8_t srcAlpha = ((*src) & 0x03) * 0x55;
const uint8_t alpha = LCD::div255((*covers++) * srcAlpha);
if (alpha == 0xFF)
{
// Solid pixel
*p = *src;
}
else if (alpha)
{
// Non-Transparent pixel
*p = mixColors(*src, *p, alpha);
}
p++;
src++;
} while (--count != 0);
}
else
{
do
{
const uint8_t srcAlpha = ((*src) & 0x03) * 0x55;
const uint8_t alpha = LCD::div255((*covers++) * srcAlpha);
if (alpha) // This can never get to max=0xFF*0xFF as totalAlpha<255
{
// Non-Transparent pixel
*p = mixColors(*src, *p, alpha);
}
p++;
src++;
} while (--count != 0);
}
}
bool PainterBGRA2222Bitmap::renderInit()
{
bitmapBGRA2222Pointer = 0;
if (bitmap.getId() == BITMAP_INVALID)
{
return false;
}
if ((currentX >= bitmapRectToFrameBuffer.width) || (currentY >= bitmapRectToFrameBuffer.height))
{
// Outside bitmap area, do not draw anything
// Consider the following instead of "return" to get a tiled image:
// currentX %= bitmapRectToFrameBuffer.width
// currentY %= bitmapRectToFrameBuffer.height
return false;
}
if (bitmap.getFormat() == Bitmap::BGRA2222)
{
bitmapBGRA2222Pointer = bitmap.getData();
if (!bitmapBGRA2222Pointer)
{
return false;
}
bitmapBGRA2222Pointer += currentX + currentY * bitmapRectToFrameBuffer.width;
return true;
}
return false;
}
bool PainterBGRA2222Bitmap::renderNext(uint8_t& red, uint8_t& green, uint8_t& blue, uint8_t& alpha)
{
if (currentX >= bitmapRectToFrameBuffer.width)
{
return false;
}
else if (bitmapBGRA2222Pointer != 0)
{
uint16_t BGRA2222 = *bitmapBGRA2222Pointer++;
red = LCD8bpp_BGRA2222::getRedFromColor(BGRA2222);
green = LCD8bpp_BGRA2222::getGreenFromColor(BGRA2222);
blue = LCD8bpp_BGRA2222::getBlueFromColor(BGRA2222);
alpha = (BGRA2222 & 0x03) * 0x55; // To get full range 0-0xFF
}
// Apply given alpha from setAlpha()
alpha = LCD::div255(alpha * painterAlpha);
return true;
}
} // namespace touchgfx

View File

@ -0,0 +1,81 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <touchgfx/widgets/canvas/PainterBW.hpp>
namespace touchgfx
{
void PainterBW::render(uint8_t* ptr, int x, int xAdjust, int y, unsigned count, const uint8_t* covers)
{
currentX = x + areaOffsetX;
currentY = y + areaOffsetY;
x += xAdjust;
unsigned char* p = ptr + (x / 8);
const uint8_t totalAlpha = LCD::div255(widgetAlpha * painterAlpha);
if (totalAlpha == 0xFF)
{
do
{
if (*covers++ >= 0x80)
{
unsigned pixel = 1 << (7 - (x % 8));
if (!painterColor)
{
*p &= ~pixel;
}
else
{
*p |= pixel;
}
}
if (((++x) % 8) == 0)
{
p++;
}
currentX++;
} while (--count);
}
else
{
do
{
if (totalAlpha * *covers++ >= 0xFF * 0x80)
{
unsigned pixel = 1 << (7 - (x % 8));
if (!painterColor)
{
*p &= ~pixel;
}
else
{
*p |= pixel;
}
}
if (((++x) % 8) == 0)
{
p++;
}
currentX++;
} while (--count);
}
}
bool PainterBW::renderNext(uint8_t& color)
{
color = painterColor;
return true;
}
} // namespace touchgfx

View File

@ -0,0 +1,210 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <touchgfx/widgets/canvas/PainterBWBitmap.hpp>
namespace touchgfx
{
void PainterBWBitmap::setBitmap(const Bitmap& bmp)
{
bitmap = bmp;
assert((bitmap.getId() == BITMAP_INVALID || bitmap.getFormat() == Bitmap::BW || bitmap.getFormat() == Bitmap::BW_RLE) && "The chosen painter only works with BW and BW_RLE bitmaps");
bitmapRectToFrameBuffer = bitmap.getRect();
DisplayTransformation::transformDisplayToFrameBuffer(bitmapRectToFrameBuffer);
}
// Found in LCD1bpp
void fillBits(uint8_t* fb, uint16_t startX, uint16_t startY, uint16_t stride, uint32_t count, uint8_t color);
void PainterBWBitmap::render(uint8_t* ptr, int x, int xAdjust, int y, unsigned count, const uint8_t* /*covers*/)
{
currentX = x + areaOffsetX;
currentY = y + areaOffsetY;
x += xAdjust;
if (!renderInit())
{
return;
}
if (currentX + (int)count > bitmapRectToFrameBuffer.width)
{
count = bitmapRectToFrameBuffer.width - currentX;
}
if (bitmap.getFormat() == Bitmap::BW_RLE)
{
while (count)
{
uint32_t length = bw_rle.getLength();
uint32_t bitsToDraw = MIN(length, (uint32_t)count);
fillBits(ptr, x, 0, 0 /* not used */, bitsToDraw, bw_rle.getColor());
x += bitsToDraw;
count -= bitsToDraw;
bw_rle.skipNext(bitsToDraw);
}
}
else
{
unsigned char* p = ptr + (x / 8);
const uint8_t* src = bitmapBWPointer + currentX / 8;
uint8_t* RESTRICT dst = p;
uint16_t srcBitX = currentX % 8; // & 7
uint16_t dstBitX = x % 8; // & 7
uint16_t remainingBits = count;
if (dstBitX > 0)
{
// Start by getting (dst-)aligned for faster transfer
uint16_t neededBits = 8 - dstBitX;
if (neededBits > remainingBits)
{
neededBits = remainingBits; // Very narrow src inside same word
}
const uint16_t availableBits = 8 - srcBitX;
uint8_t mask = (1u << neededBits) - 1u;
const uint8_t dstShift = static_cast<uint8_t>(8u - (dstBitX + neededBits));
mask <<= dstShift;
uint8_t word = *src;
if (availableBits > neededBits)
{
word >>= availableBits - neededBits;
}
else if (availableBits < neededBits)
{
// Get the last required bits from src[1]
word <<= neededBits - availableBits;
word |= src[1] >> (8u - (neededBits - availableBits));
}
word <<= dstShift;
*dst = (*dst & ~mask) | (word & mask);
srcBitX = (srcBitX + neededBits) % 8; // & 7
if (availableBits <= neededBits)
{
src++;
}
dst++;
remainingBits -= neededBits;
}
// dstX is now word aligned (or we have transferred everything of a narrow image and remainingBits==0)
if (remainingBits >= 8)
{
uint16_t bytesPerLine = remainingBits / 8;
if (srcBitX == 0)
{
HAL::getInstance()->blockCopy(dst, src, bytesPerLine);
src += bytesPerLine;
dst += bytesPerLine;
}
else
{
uint16_t _remainingBits = remainingBits;
remainingBits = _remainingBits;
while (remainingBits >= 8)
{
uint8_t word = *src++;
word <<= srcBitX;
word |= (*src) >> (8 - srcBitX);
*dst++ = word;
remainingBits -= 8;
}
}
remainingBits %= 8; // &= 7
}
// Take the last bits, again we need to mask dst
if (remainingBits > 0)
{
uint8_t word = *src;
if (srcBitX != 0)
{
word <<= srcBitX;
word |= src[1] >> (8u - srcBitX);
}
const uint8_t mask = ((1u << remainingBits) - 1u) << (8u - remainingBits);
*dst = (*dst & ~mask) | (word & mask);
}
}
}
bool PainterBWBitmap::renderInit()
{
bw_rle = 0; // Used to remember if format is BW or BW_RLE
bitmapBWPointer = 0;
if (bitmap.getId() == BITMAP_INVALID)
{
return false;
}
if (currentX >= bitmapRectToFrameBuffer.width || currentY >= bitmapRectToFrameBuffer.height)
{
// Outside bitmap area, do not draw anything
return false;
}
// Common for BW and BW_RLE
bitmapBWPointer = (const uint8_t*)bitmap.getData();
if (!bitmapBWPointer)
{
return false;
}
if (bitmap.getFormat() == Bitmap::BW_RLE)
{
bw_rle.init(bitmapBWPointer);
uint32_t skip = (int32_t)currentY * (int32_t)bitmapRectToFrameBuffer.width + (int32_t)currentX;
bw_rle.skipNext(skip);
return true;
}
if (bitmap.getFormat() == Bitmap::BW)
{
bitmapBWPointer += currentY * ((bitmapRectToFrameBuffer.width + 7) / 8);
return true;
}
return false;
}
bool PainterBWBitmap::renderNext(uint8_t& color)
{
if (currentX >= bitmapRectToFrameBuffer.width || bitmapBWPointer == 0)
{
return false;
}
if (bitmap.getFormat() == Bitmap::BW_RLE)
{
color = bw_rle.getColor();
bw_rle.skipNext(1);
}
else
{
const uint8_t* src = bitmapBWPointer + currentX / 8;
color = ((*src) >> (7 - (currentX % 8))) & 1;
}
return true;
}
} // namespace touchgfx

View File

@ -0,0 +1,75 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <platform/driver/lcd/LCD2bpp.hpp>
#include <touchgfx/Color.hpp>
#include <touchgfx/widgets/canvas/PainterGRAY2.hpp>
namespace touchgfx
{
void PainterGRAY2::render(uint8_t* ptr,
int x,
int xAdjust,
int y,
unsigned count,
const uint8_t* covers)
{
currentX = x + areaOffsetX;
currentY = y + areaOffsetY;
x += xAdjust;
const uint8_t totalAlpha = LCD::div255(widgetAlpha * painterAlpha);
if (totalAlpha == 0xFF)
{
do
{
const uint8_t alpha = *covers++;
if (alpha == 0xFF)
{
// Render a solid pixel
LCD2bpp::setPixel(ptr, x, painterGray);
}
else
{
const uint8_t ialpha = 0xFF - alpha;
const uint8_t p_gray = LCD2bpp::getPixel(ptr, x);
LCD2bpp::setPixel(ptr, x, LCD::div255((painterGray * alpha + p_gray * ialpha) * 0x55) >> 6);
}
currentX++;
x++;
} while (--count != 0);
}
else if (totalAlpha != 0)
{
do
{
const uint8_t alpha = LCD::div255((*covers++) * totalAlpha);
const uint8_t ialpha = 0xFF - alpha;
const uint8_t p_gray = LCD2bpp::getPixel(ptr, x);
LCD2bpp::setPixel(ptr, x, LCD::div255((painterGray * alpha + p_gray * ialpha) * 0x55) >> 6);
currentX++;
x++;
} while (--count != 0);
}
}
bool PainterGRAY2::renderNext(uint8_t& gray, uint8_t& alpha)
{
gray = painterGray;
alpha = painterAlpha;
return true;
}
} // namespace touchgfx

View File

@ -0,0 +1,185 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <platform/driver/lcd/LCD2bpp.hpp>
#include <touchgfx/widgets/canvas/PainterGRAY2Bitmap.hpp>
namespace touchgfx
{
void PainterGRAY2Bitmap::setBitmap(const Bitmap& bmp)
{
bitmap = bmp;
assert((bitmap.getId() == BITMAP_INVALID || bitmap.getFormat() == Bitmap::GRAY2) && "The chosen painter only works with GRAY2 bitmaps");
bitmapRectToFrameBuffer = bitmap.getRect();
DisplayTransformation::transformDisplayToFrameBuffer(bitmapRectToFrameBuffer);
}
void PainterGRAY2Bitmap::render(uint8_t* ptr, int x, int xAdjust, int y, unsigned count, const uint8_t* covers)
{
currentX = x + areaOffsetX;
currentY = y + areaOffsetY;
x += xAdjust;
if (!renderInit())
{
return;
}
if (currentX + (int)count > bitmapRectToFrameBuffer.width)
{
count = bitmapRectToFrameBuffer.width - currentX;
}
const uint8_t totalAlpha = LCD::div255(widgetAlpha * painterAlpha);
if (bitmapAlphaPointer)
{
if (totalAlpha == 0xFF)
{
do
{
const uint8_t gray = LCD2bpp::getPixel(bitmapGRAY2Pointer, currentX);
const uint8_t alpha = LCD::div255((*covers++) * (LCD2bpp::getPixel(bitmapAlphaPointer, currentX) * 0x55));
if (alpha == 0xFF)
{
// Render a solid pixel
LCD2bpp::setPixel(ptr, x, gray);
}
else
{
const uint8_t ialpha = 0xFF - alpha;
const uint8_t p_gray = LCD2bpp::getPixel(ptr, x);
LCD2bpp::setPixel(ptr, x, LCD::div255((gray * alpha + p_gray * ialpha) * 0x55) >> 6);
}
currentX++;
x++;
} while (--count != 0);
}
else
{
do
{
const uint8_t gray = LCD2bpp::getPixel(bitmapGRAY2Pointer, currentX);
const uint8_t alpha = LCD::div255((*covers++) * LCD::div255(totalAlpha * (LCD2bpp::getPixel(bitmapAlphaPointer, currentX) * 0x55)));
const uint8_t ialpha = 0xFF - alpha;
const uint8_t p_gray = LCD2bpp::getPixel(ptr, x);
LCD2bpp::setPixel(ptr, x, LCD::div255((gray * alpha + p_gray * ialpha) * 0x55) >> 6);
currentX++;
x++;
} while (--count != 0);
}
}
else
{
if (totalAlpha == 255)
{
do
{
const uint8_t gray = LCD2bpp::getPixel(bitmapGRAY2Pointer, currentX);
const uint8_t alpha = *covers++;
if (alpha == 255)
{
// Render a solid pixel
LCD2bpp::setPixel(ptr, x, gray);
}
else
{
const uint8_t ialpha = 0xFF - alpha;
const uint8_t p_gray = LCD2bpp::getPixel(ptr, x);
LCD2bpp::setPixel(ptr, x, LCD::div255((gray * alpha + p_gray * ialpha) * 0x55) >> 6);
}
currentX++;
x++;
} while (--count != 0);
}
else
{
do
{
const uint8_t gray = LCD2bpp::getPixel(bitmapGRAY2Pointer, currentX);
const uint8_t alpha = LCD::div255((*covers++) * totalAlpha);
const uint8_t ialpha = 0xFF - alpha;
const uint8_t p_gray = LCD2bpp::getPixel(ptr, x);
LCD2bpp::setPixel(ptr, x, LCD::div255((gray * alpha + p_gray * ialpha) * 0x55) >> 6);
currentX++;
x++;
} while (--count != 0);
}
}
}
bool PainterGRAY2Bitmap::renderInit()
{
bitmapGRAY2Pointer = 0;
bitmapAlphaPointer = 0;
if (bitmap.getId() == BITMAP_INVALID)
{
return false;
}
if ((currentX >= bitmapRectToFrameBuffer.width) || (currentY >= bitmapRectToFrameBuffer.height))
{
// Outside bitmap area, do not draw anything
return false;
}
if (bitmap.getFormat() == Bitmap::GRAY2)
{
bitmapGRAY2Pointer = (const uint8_t*)bitmap.getData();
if (!bitmapGRAY2Pointer)
{
return false;
}
bitmapGRAY2Pointer += currentY * ((bitmapRectToFrameBuffer.width + 3) / 4);
bitmapAlphaPointer = (const uint8_t*)bitmap.getExtraData();
if (bitmapAlphaPointer)
{
bitmapAlphaPointer += currentY * ((bitmapRectToFrameBuffer.width + 3) / 4);
}
return true;
}
return false;
}
bool PainterGRAY2Bitmap::renderNext(uint8_t& gray, uint8_t& alpha)
{
if (currentX >= bitmapRectToFrameBuffer.width)
{
return false;
}
if (bitmapGRAY2Pointer != 0)
{
gray = LCD2bpp::getPixel(bitmapGRAY2Pointer, currentX);
if (bitmapAlphaPointer)
{
alpha = LCD2bpp::getPixel(bitmapAlphaPointer, currentX);
alpha *= 0x55; // Upscale from 00-03 to 00-FF
}
else
{
alpha = 0xFF; // No alpha per pixel in the image, it is solid
}
}
// Apply given alpha from setAlpha()
alpha = LCD::div255(alpha * painterAlpha);
return true;
}
} // namespace touchgfx

View File

@ -0,0 +1,68 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <platform/driver/lcd/LCD4bpp.hpp>
#include <touchgfx/Color.hpp>
#include <touchgfx/widgets/canvas/PainterGRAY4.hpp>
namespace touchgfx
{
void PainterGRAY4::render(uint8_t* ptr, int x, int xAdjust, int y, unsigned count, const uint8_t* covers)
{
currentX = x + areaOffsetX;
currentY = y + areaOffsetY;
x += xAdjust;
const uint8_t totalAlpha = LCD::div255(widgetAlpha * painterAlpha);
if (totalAlpha == 0xFF)
{
do
{
const uint8_t alpha = *covers++;
if (alpha == 0xFF) // max alpha=0xFF on "*covers" and max alpha=0xFF on "widgetAlpha"
{
// Render a solid pixel
LCD4bpp::setPixel(ptr, x, painterGray);
}
else
{
const uint8_t ialpha = 0xFF - alpha;
const uint8_t p_gray = LCD4bpp::getPixel(ptr, x);
LCD4bpp::setPixel(ptr, x, LCD::div255((painterGray * alpha + p_gray * ialpha) * 0x11) >> 4);
}
currentX++;
x++;
} while (--count != 0);
}
else if (totalAlpha != 0)
{
do
{
const uint8_t alpha = LCD::div255((*covers++) * totalAlpha);
const uint8_t ialpha = 0xFF - alpha;
const uint8_t p_gray = LCD4bpp::getPixel(ptr, x);
LCD4bpp::setPixel(ptr, x, LCD::div255((painterGray * alpha + p_gray * ialpha) * 0x11) >> 4);
currentX++;
x++;
} while (--count != 0);
}
}
bool PainterGRAY4::renderNext(uint8_t& gray, uint8_t& alpha)
{
gray = painterGray;
alpha = painterAlpha;
return true;
}
} // namespace touchgfx

View File

@ -0,0 +1,186 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <platform/driver/lcd/LCD4bpp.hpp>
#include <touchgfx/widgets/canvas/PainterGRAY4Bitmap.hpp>
namespace touchgfx
{
void PainterGRAY4Bitmap::setBitmap(const Bitmap& bmp)
{
bitmap = bmp;
assert((bitmap.getId() == BITMAP_INVALID || bitmap.getFormat() == Bitmap::GRAY4) && "The chosen painter only works with GRAY4 bitmaps");
bitmapRectToFrameBuffer = bitmap.getRect();
DisplayTransformation::transformDisplayToFrameBuffer(bitmapRectToFrameBuffer);
}
void PainterGRAY4Bitmap::render(uint8_t* ptr, int x, int xAdjust, int y, unsigned count, const uint8_t* covers)
{
currentX = x + areaOffsetX;
currentY = y + areaOffsetY;
x += xAdjust;
if (!renderInit())
{
return;
}
if (currentX + (int)count > bitmapRectToFrameBuffer.width)
{
count = bitmapRectToFrameBuffer.width - currentX;
}
const uint8_t totalAlpha = LCD::div255(widgetAlpha * painterAlpha);
// Get alpha data (GRAY4 format)
if (bitmapAlphaPointer)
{
if (totalAlpha == 0xFF)
{
do
{
const uint8_t gray = LCD4bpp::getPixel(bitmapGRAY4Pointer, currentX);
const uint8_t alpha = LCD::div255((*covers++) * (LCD4bpp::getPixel(bitmapAlphaPointer, currentX) * 0x11));
if (alpha == 0xFF)
{
// Render a solid pixel
LCD4bpp::setPixel(ptr, x, gray);
}
else
{
const uint8_t ialpha = 0xFF - alpha;
const uint8_t p_gray = LCD4bpp::getPixel(ptr, x);
LCD4bpp::setPixel(ptr, x, LCD::div255((gray * alpha + p_gray * ialpha) * 0x11) >> 4);
}
currentX++;
x++;
} while (--count != 0);
}
else
{
do
{
const uint8_t gray = LCD4bpp::getPixel(bitmapGRAY4Pointer, currentX);
const uint8_t alpha = LCD::div255((*covers++) * LCD::div255(totalAlpha * (LCD4bpp::getPixel(bitmapAlphaPointer, currentX) * 0x11)));
const uint8_t ialpha = 0xFF - alpha;
const uint8_t p_gray = LCD4bpp::getPixel(ptr, x);
LCD4bpp::setPixel(ptr, x, LCD::div255((gray * alpha + p_gray * ialpha) * 0x11) >> 4);
currentX++;
x++;
} while (--count != 0);
}
}
else
{
if (totalAlpha == 255)
{
do
{
const uint8_t gray = LCD4bpp::getPixel(bitmapGRAY4Pointer, currentX);
const uint8_t alpha = *covers++;
if (alpha == 0xFF)
{
// Render a solid pixel
LCD4bpp::setPixel(ptr, x, gray);
}
else
{
const uint8_t ialpha = 0xFF - alpha;
const uint8_t p_gray = LCD4bpp::getPixel(ptr, x);
LCD4bpp::setPixel(ptr, x, LCD::div255((gray * alpha + p_gray * ialpha) * 0x11) >> 4);
}
currentX++;
x++;
} while (--count != 0);
}
else
{
do
{
const uint8_t gray = LCD4bpp::getPixel(bitmapGRAY4Pointer, currentX);
const uint8_t alpha = LCD::div255((*covers++) * totalAlpha);
const uint8_t ialpha = 0xFF - alpha;
const uint8_t p_gray = LCD4bpp::getPixel(ptr, x);
LCD4bpp::setPixel(ptr, x, LCD::div255((gray * alpha + p_gray * ialpha) * 0x11) >> 4);
currentX++;
x++;
} while (--count != 0);
}
}
}
bool PainterGRAY4Bitmap::renderInit()
{
bitmapGRAY4Pointer = 0;
bitmapAlphaPointer = 0;
if (bitmap.getId() == BITMAP_INVALID)
{
return false;
}
if ((currentX >= bitmapRectToFrameBuffer.width) || (currentY >= bitmapRectToFrameBuffer.height))
{
// Outside bitmap area, do not draw anything
return false;
}
if (bitmap.getFormat() == Bitmap::GRAY4)
{
bitmapGRAY4Pointer = (const uint8_t*)bitmap.getData();
if (!bitmapGRAY4Pointer)
{
return false;
}
bitmapGRAY4Pointer += currentY * ((bitmapRectToFrameBuffer.width + 1) / 2);
bitmapAlphaPointer = (const uint8_t*)bitmap.getExtraData();
if (bitmapAlphaPointer)
{
bitmapAlphaPointer += currentY * ((bitmapRectToFrameBuffer.width + 1) / 2);
}
return true;
}
return false;
}
bool PainterGRAY4Bitmap::renderNext(uint8_t& gray, uint8_t& alpha)
{
if (currentX >= bitmapRectToFrameBuffer.width)
{
return false;
}
if (bitmapGRAY4Pointer != 0)
{
gray = LCD4bpp::getPixel(bitmapGRAY4Pointer, currentX);
if (bitmapAlphaPointer)
{
alpha = LCD4bpp::getPixel(bitmapAlphaPointer, currentX);
alpha |= alpha << 4; // Upscale from 0-15 to 0-255
}
else
{
alpha = 0xFF; // No alpha per pixel in the image, it is solid
}
}
// Apply given alpha from setAlpha()
alpha = LCD::div255(alpha * painterAlpha);
return true;
}
} // namespace touchgfx

View File

@ -0,0 +1,67 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <touchgfx/Color.hpp>
#include <touchgfx/widgets/canvas/PainterRGB565.hpp>
namespace touchgfx
{
void PainterRGB565::render(uint8_t* ptr, int x, int xAdjust, int /*y*/, unsigned count, const uint8_t* covers)
{
uint16_t* p = reinterpret_cast<uint16_t*>(ptr) + (x + xAdjust);
const uint8_t totalAlpha = LCD::div255(widgetAlpha * painterAlpha);
if (totalAlpha == 0xFF)
{
do
{
const uint8_t alpha = *covers++;
if (alpha == 0xFF)
{
*p = painterColor;
}
else
{
*p = mixColors(painterRed, painterGreen, painterBlue, *p, alpha);
}
p++;
} while (--count != 0);
}
else
{
do
{
const uint8_t alpha = LCD::div255((*covers++) * totalAlpha);
if (alpha == 0xFF)
{
*p = painterColor;
}
else
{
*p = mixColors(painterRed, painterGreen, painterBlue, *p, alpha);
}
p++;
} while (--count != 0);
}
}
bool PainterRGB565::renderNext(uint8_t& red, uint8_t& green, uint8_t& blue, uint8_t& alpha)
{
red = painterRed >> 8;
green = painterGreen >> 3;
blue = painterBlue << 3;
alpha = painterAlpha;
return true;
}
} // namespace touchgfx

View File

@ -0,0 +1,258 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <touchgfx/widgets/canvas/PainterRGB565Bitmap.hpp>
namespace touchgfx
{
void PainterRGB565Bitmap::setBitmap(const Bitmap& bmp)
{
bitmap = bmp;
assert((bitmap.getId() == BITMAP_INVALID || bitmap.getFormat() == Bitmap::RGB565 || bitmap.getFormat() == Bitmap::ARGB8888) && "The chosen painter only works with RGB565 and ARGB8888 bitmaps");
bitmapRectToFrameBuffer = bitmap.getRect();
DisplayTransformation::transformDisplayToFrameBuffer(bitmapRectToFrameBuffer);
}
void PainterRGB565Bitmap::render(uint8_t* ptr, int x, int xAdjust, int y, unsigned count, const uint8_t* covers)
{
uint16_t* p = reinterpret_cast<uint16_t*>(ptr) + (x + xAdjust);
currentX = x + areaOffsetX;
currentY = y + areaOffsetY;
if (!renderInit())
{
return;
}
if (currentX + (int)count > bitmapRectToFrameBuffer.width)
{
count = bitmapRectToFrameBuffer.width - currentX;
}
const uint8_t totalAlpha = LCD::div255(widgetAlpha * painterAlpha);
if (bitmap.getFormat() == Bitmap::RGB565)
{
const uint16_t* src = bitmapRGB565Pointer;
const uint8_t* srcAlpha = bitmapAlphaPointer;
if (srcAlpha)
{
if (totalAlpha == 0xFF)
{
do
{
const uint8_t alpha = LCD::div255((*covers++) * (*srcAlpha++));
if (alpha == 0xFF)
{
// Solid pixel
*p = *src;
}
else if (alpha)
{
// Non-Transparent pixel
*p = mixColors(*src, *p, alpha);
}
p++;
src++;
} while (--count != 0);
}
else
{
do
{
const uint8_t alpha = LCD::div255((*covers++) * LCD::div255((*srcAlpha++) * totalAlpha));
if (alpha) // This can never get to max=0XFF as totalAlpha<0xFF
{
// Non-Transparent pixel
*p = mixColors(*src, *p, alpha);
}
p++;
src++;
} while (--count != 0);
}
}
else
{
if (totalAlpha == 0xFF)
{
do
{
//use alpha from covers directly
const uint8_t alpha = *covers++;
if (alpha == 0xFF)
{
// Solid pixel
*p = *src;
}
else
{
// Non-Transparent pixel
*p = mixColors(*src, *p, alpha);
}
p++;
src++;
} while (--count != 0);
}
else
{
do
{
const uint8_t alpha = LCD::div255((*covers++) * totalAlpha);
*p = mixColors(*src, *p, alpha);
p++;
src++;
} while (--count != 0);
}
}
}
else if (bitmap.getFormat() == Bitmap::ARGB8888)
{
const uint32_t* src = bitmapARGB8888Pointer;
if (totalAlpha == 0xFF)
{
uint32_t newpix;
do
{
const uint8_t srcAlpha = (*src) >> 24;
const uint8_t alpha = LCD::div255((*covers++) * srcAlpha);
newpix = *src;
if (alpha == 0xFF)
{
// Solid pixel
*p = ((newpix >> 8) & RMASK) | ((newpix >> 5) & GMASK) | ((newpix >> 3) & BMASK);
}
else if (alpha)
{
// Non-Transparent pixel
*p = mixColors((newpix >> 8) & RMASK, (newpix >> 5) & GMASK, (newpix >> 3) & BMASK, *p, alpha);
}
p++;
src++;
} while (--count != 0);
}
else
{
uint32_t newpix;
do
{
const uint8_t srcAlpha = (*src) >> 24;
const uint8_t alpha = LCD::div255((*covers++) * LCD::div255(srcAlpha * totalAlpha));
if (alpha)
{
// Non-Transparent pixel
newpix = *src;
*p = mixColors((newpix >> 8) & RMASK, (newpix >> 5) & GMASK, (newpix >> 3) & BMASK, *p, alpha);
}
p++;
src++;
} while (--count != 0);
}
}
}
bool PainterRGB565Bitmap::renderInit()
{
bitmapARGB8888Pointer = 0;
bitmapRGB565Pointer = 0;
bitmapAlphaPointer = 0;
if (bitmap.getId() == BITMAP_INVALID)
{
return false;
}
if ((currentX >= bitmapRectToFrameBuffer.width) || (currentY >= bitmapRectToFrameBuffer.height))
{
// Outside bitmap area, do not draw anything
// Consider the following instead of "return" to get a tiled image:
// currentX %= bitmapRectToFrameBuffer.width
// currentY %= bitmapRectToFrameBuffer.height
return false;
}
if (bitmap.getFormat() == Bitmap::ARGB8888)
{
bitmapARGB8888Pointer = (const uint32_t*)bitmap.getData();
if (!bitmapARGB8888Pointer)
{
return false;
}
bitmapARGB8888Pointer += currentX + currentY * bitmapRectToFrameBuffer.width;
return true;
}
if (bitmap.getFormat() == Bitmap::RGB565)
{
bitmapRGB565Pointer = (const uint16_t*)bitmap.getData();
if (!bitmapRGB565Pointer)
{
return false;
}
bitmapRGB565Pointer += currentX + currentY * bitmapRectToFrameBuffer.width;
// Get alpha data (RGB565 format)
bitmapAlphaPointer = (const uint8_t*)bitmap.getExtraData();
if (bitmapAlphaPointer)
{
bitmapAlphaPointer += currentX + currentY * bitmapRectToFrameBuffer.width;
}
return true;
}
return false;
}
bool PainterRGB565Bitmap::renderNext(uint8_t& red, uint8_t& green, uint8_t& blue, uint8_t& alpha)
{
if (currentX >= bitmapRectToFrameBuffer.width)
{
return false;
}
if (bitmapARGB8888Pointer != 0)
{
uint32_t argb8888 = *bitmapARGB8888Pointer++;
alpha = (argb8888 >> 24) & 0xFF;
red = (argb8888 >> 16) & 0xF8;
red |= red >> 5; // To get full range 0-0xFF, not just 0-0xF8
green = (argb8888 >> 8) & 0xFC;
green |= green >> 6; // To get full range 0-0xFF, not just 0-0xFC
blue = argb8888 & 0xF8;
blue |= blue >> 5; // To get full range 0-0xFF, not just 0-0xF8
}
else if (bitmapRGB565Pointer != 0)
{
uint16_t rgb565 = *bitmapRGB565Pointer++;
red = (rgb565 & RMASK) >> 8;
red |= red >> 5; // To get full range 0-0xFF, not just 0-0xF8
green = (rgb565 & GMASK) >> 3;
green |= green >> 6; // To get full range 0-0xFF, not just 0-0xFC
blue = (rgb565 & BMASK) << 3;
blue |= blue >> 5; // To get full range 0-0xFF, not just 0-0xF8
if (bitmapAlphaPointer)
{
alpha = *bitmapAlphaPointer++;
}
else
{
alpha = 0xFF; // No alpha per pixel in the image, it is solid
}
}
// Apply given alpha from setAlpha()
alpha = LCD::div255(alpha * painterAlpha);
return true;
}
} // namespace touchgfx

View File

@ -0,0 +1,257 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <touchgfx/widgets/canvas/PainterRGB565L8Bitmap.hpp>
namespace touchgfx
{
void PainterRGB565L8Bitmap::setBitmap(const Bitmap& bmp)
{
bitmap = bmp;
assert((bitmap.getId() == BITMAP_INVALID || bitmap.getFormat() == Bitmap::L8) && "The chosen painter only works with appropriate L8 bitmaps");
bitmapRectToFrameBuffer = bitmap.getRect();
DisplayTransformation::transformDisplayToFrameBuffer(bitmapRectToFrameBuffer);
}
void PainterRGB565L8Bitmap::render(uint8_t* ptr, int x, int xAdjust, int y, unsigned count, const uint8_t* covers)
{
uint16_t* p = reinterpret_cast<uint16_t*>(ptr) + (x + xAdjust);
currentX = x + areaOffsetX;
currentY = y + areaOffsetY;
if (!renderInit())
{
return;
}
if (currentX + (int)count > bitmapRectToFrameBuffer.width)
{
count = bitmapRectToFrameBuffer.width - currentX;
}
const uint8_t totalAlpha = LCD::div255(widgetAlpha * painterAlpha);
if ((Bitmap::ClutFormat)((const uint16_t*)bitmapExtraPointer)[-2] == Bitmap::CLUT_FORMAT_L8_RGB565)
{
const uint8_t* src = bitmapPointer;
if (totalAlpha == 0xFF)
{
do
{
//use alpha from covers directly
const uint8_t alpha = *covers++;
if (alpha == 0xFF)
{
// Solid pixel
*p = ((const uint16_t*)bitmapExtraPointer)[*src];
}
else
{
// Non-Transparent pixel
*p = mixColors(((const uint16_t*)bitmapExtraPointer)[*src], *p, alpha);
}
p++;
src++;
} while (--count != 0);
}
else
{
do
{
const uint8_t alpha = LCD::div255((*covers++) * totalAlpha);
*p = mixColors(((const uint16_t*)bitmapExtraPointer)[*src], *p, alpha);
p++;
src++;
} while (--count != 0);
}
}
else if ((Bitmap::ClutFormat)((const uint16_t*)bitmapExtraPointer)[-2] == Bitmap::CLUT_FORMAT_L8_RGB888)
{
if (totalAlpha == 0xFF)
{
do
{
const uint8_t* src = &bitmapExtraPointer[*bitmapPointer++ * 3];
// Use alpha from covers directly
const uint8_t alpha = *covers++;
const uint8_t blue = *src++;
const uint8_t green = *src++;
const uint8_t red = *src;
if (alpha == 0xFF)
{
// Solid pixel
*p++ = ((red << 8) & RMASK) | ((green << 3) & GMASK) | ((blue >> 3) & BMASK);
}
else
{
const uint8_t ialpha = 0xFF - alpha;
const uint16_t bufpix = *p;
uint8_t fbr = (bufpix & RMASK) >> 11;
uint8_t fbg = (bufpix & GMASK) >> 5;
uint8_t fbb = bufpix & BMASK;
fbr = (fbr * 527 + 23) >> 6;
fbg = (fbg * 259 + 33) >> 6;
fbb = (fbb * 527 + 23) >> 6;
*p++ = ((LCD::div255(red * alpha + fbr * ialpha) << 8) & RMASK) | ((LCD::div255(green * alpha + fbg * ialpha) << 3) & GMASK) | ((LCD::div255(blue * alpha + fbb * ialpha) >> 3) & BMASK);
}
} while (--count != 0);
}
else
{
do
{
const uint8_t* src = &bitmapExtraPointer[*bitmapPointer++ * 3];
const uint8_t blue = *src++;
const uint8_t green = *src++;
const uint8_t red = *src;
const uint8_t alpha = LCD::div255((*covers++) * totalAlpha);
const uint8_t ialpha = 0xFF - alpha;
const uint16_t bufpix = *p;
uint8_t fbr = (bufpix & RMASK) >> 11;
uint8_t fbg = (bufpix & GMASK) >> 5;
uint8_t fbb = bufpix & BMASK;
fbr = (fbr * 527 + 23) >> 6;
fbg = (fbg * 259 + 33) >> 6;
fbb = (fbb * 527 + 23) >> 6;
*p++ = ((LCD::div255(red * alpha + fbr * ialpha) << 8) & RMASK) | ((LCD::div255(green * alpha + fbg * ialpha) << 3) & GMASK) | ((LCD::div255(blue * alpha + fbb * ialpha) >> 3) & BMASK);
} while (--count != 0);
}
}
else // Bitmap::CLUT_FORMAT_L8_ARGB8888
{
const uint8_t* src = bitmapPointer;
if (totalAlpha == 0xFF)
{
do
{
const uint32_t newpix = ((const uint32_t*)bitmapExtraPointer)[*src];
const uint8_t srcAlpha = newpix >> 24;
const uint8_t alpha = LCD::div255((*covers++) * srcAlpha);
if (alpha == 0xFF)
{
// Solid pixel
*p = ((newpix >> 8) & RMASK) | ((newpix >> 5) & GMASK) | ((newpix >> 3) & BMASK);
}
else if (alpha)
{
// Non-Transparent pixel
*p = mixColors((newpix >> 8) & RMASK, (newpix >> 5) & GMASK, (newpix >> 3) & BMASK, *p, alpha);
}
p++;
src++;
} while (--count != 0);
}
else
{
do
{
const uint32_t newpix = ((const uint32_t*)bitmapExtraPointer)[*src];
const uint8_t srcAlpha = newpix >> 24;
const uint8_t alpha = LCD::div255((*covers++) * LCD::div255(srcAlpha * totalAlpha));
if (alpha)
{
// Non-Transparent pixel
*p = mixColors((newpix >> 8) & RMASK, (newpix >> 5) & GMASK, (newpix >> 3) & BMASK, *p, alpha);
}
p++;
src++;
} while (--count != 0);
}
}
}
bool PainterRGB565L8Bitmap::renderInit()
{
bitmapPointer = 0;
bitmapExtraPointer = 0;
if (bitmap.getId() == BITMAP_INVALID)
{
return false;
}
if ((currentX >= bitmapRectToFrameBuffer.width) || (currentY >= bitmapRectToFrameBuffer.height))
{
// Outside bitmap area, do not draw anything
// Consider the following instead of "return" to get a tiled image:
// currentX %= bitmapRectToFrameBuffer.width
// currentY %= bitmapRectToFrameBuffer.height
return false;
}
if (bitmap.getFormat() == Bitmap::L8)
{
bitmapPointer = bitmap.getData();
if (!bitmapPointer)
{
return false;
}
bitmapPointer += currentX + currentY * bitmapRectToFrameBuffer.width;
bitmapExtraPointer = bitmap.getExtraData();
assert((bitmapExtraPointer != 0 && (*(const uint16_t*)bitmapExtraPointer == Bitmap::CLUT_FORMAT_L8_RGB565 || *(const uint16_t*)bitmapExtraPointer == Bitmap::CLUT_FORMAT_L8_ARGB8888 || *(const uint16_t*)bitmapExtraPointer == Bitmap::CLUT_FORMAT_L8_RGB888)));
bitmapExtraPointer += 4; // Skip header
return true;
}
return false;
}
bool PainterRGB565L8Bitmap::renderNext(uint8_t& red, uint8_t& green, uint8_t& blue, uint8_t& alpha)
{
if (currentX >= bitmapRectToFrameBuffer.width)
{
return false;
}
if ((Bitmap::ClutFormat)((const uint16_t*)bitmapExtraPointer)[-2] == Bitmap::CLUT_FORMAT_L8_RGB565)
{
const uint16_t* clut = ((const uint16_t*)bitmapExtraPointer) + (*bitmapPointer++);
uint16_t rgb565 = *clut;
red = (rgb565 & RMASK) >> 8;
red |= red >> 5; // To get full range 0-0xFF, not just 0-0xF8
green = (rgb565 & GMASK) >> 3;
green |= green >> 6; // To get full range 0-0xFF, not just 0-0xFC
blue = (rgb565 & BMASK) << 3;
blue |= (blue >> 5); // To get full range 0-0xFF, not just 0-0xF8
alpha = 0xFF;
}
else if ((Bitmap::ClutFormat)((const uint16_t*)bitmapExtraPointer)[-2] == Bitmap::CLUT_FORMAT_L8_RGB888)
{
const uint8_t* clut = ((const uint8_t*)bitmapExtraPointer) + (*bitmapPointer++) * 3;
alpha = 0xFF;
blue = *clut++ & 0xF8;
blue |= (blue >> 5); // To get full range 0-0xFF, not just 0-0xF8
green = *clut++ & 0xFC;
green |= green >> 6; // To get full range 0-0xFF, not just 0-0xFC
red = *clut & 0xF8;
red |= red >> 5; // To get full range 0-0xFF, not just 0-0xF8
}
else // Bitmap::CLUT_FORMAT_L8_ARGB8888
{
const uint32_t* clut = ((const uint32_t*)bitmapExtraPointer) + (*bitmapPointer++);
uint32_t argb8888 = *(const uint32_t*)clut;
alpha = (argb8888 >> 24) & 0xFF;
red = (argb8888 >> 16) & 0xF8;
red |= red >> 5; // To get full range 0-0xFF, not just 0-0xF8
green = (argb8888 >> 8) & 0xFC;
green |= green >> 6; // To get full range 0-0xFF, not just 0-0xFC
blue = argb8888 & 0xF8;
blue |= (blue >> 5); // To get full range 0-0xFF, not just 0-0xF8
}
// Apply given alpha from setAlpha()
alpha = LCD::div255(alpha * painterAlpha);
return true;
}
} // namespace touchgfx

View File

@ -0,0 +1,72 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <touchgfx/widgets/canvas/PainterRGB888.hpp>
namespace touchgfx
{
void PainterRGB888::render(uint8_t* ptr, int x, int xAdjust, int /*y*/, unsigned count, const uint8_t* covers)
{
uint8_t* p = reinterpret_cast<uint8_t*>(ptr) + ((x + xAdjust) * 3);
uint8_t pByte;
const uint8_t totalAlpha = LCD::div255(widgetAlpha * painterAlpha);
if (totalAlpha == 0xFF)
{
do
{
const uint8_t alpha = *covers++;
if (alpha == 0xFF)
{
*p++ = painterBlue;
*p++ = painterGreen;
*p++ = painterRed;
}
else
{
const uint8_t ialpha = 0xFF - alpha;
pByte = *p;
*p++ = LCD::div255(painterBlue * alpha + pByte * ialpha);
pByte = *p;
*p++ = LCD::div255(painterGreen * alpha + pByte * ialpha);
pByte = *p;
*p++ = LCD::div255(painterRed * alpha + pByte * ialpha);
}
} while (--count != 0);
}
else if (totalAlpha != 0)
{
do
{
const uint8_t alpha = LCD::div255(*covers++ * totalAlpha);
const uint8_t ialpha = 0xFF - alpha;
pByte = *p;
*p++ = LCD::div255(painterBlue * alpha + pByte * ialpha);
pByte = *p;
*p++ = LCD::div255(painterGreen * alpha + pByte * ialpha);
pByte = *p;
*p++ = LCD::div255(painterRed * alpha + pByte * ialpha);
} while (--count != 0);
}
}
bool PainterRGB888::renderNext(uint8_t& red, uint8_t& green, uint8_t& blue, uint8_t& alpha)
{
red = painterRed;
green = painterGreen;
blue = painterBlue;
alpha = painterAlpha;
return true;
}
} // namespace touchgfx

View File

@ -0,0 +1,222 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <touchgfx/widgets/canvas/PainterRGB888Bitmap.hpp>
namespace touchgfx
{
void PainterRGB888Bitmap::setBitmap(const Bitmap& bmp)
{
bitmap = bmp;
assert((bitmap.getId() == BITMAP_INVALID || bitmap.getFormat() == Bitmap::RGB888 || bitmap.getFormat() == Bitmap::ARGB8888) && "The chosen painter only works with RGB888 and ARGB8888 bitmaps");
bitmapRectToFrameBuffer = bitmap.getRect();
DisplayTransformation::transformDisplayToFrameBuffer(bitmapRectToFrameBuffer);
}
void PainterRGB888Bitmap::render(uint8_t* ptr, int x, int xAdjust, int y, unsigned count, const uint8_t* covers)
{
uint8_t* p = ptr + ((x + xAdjust) * 3);
currentX = x + areaOffsetX;
currentY = y + areaOffsetY;
if (!renderInit())
{
return;
}
if (currentX + (int)count > bitmapRectToFrameBuffer.width)
{
count = bitmapRectToFrameBuffer.width - currentX;
}
const uint8_t totalAlpha = LCD::div255(widgetAlpha * painterAlpha);
if (bitmapRGB888Pointer)
{
const uint8_t* src = bitmapRGB888Pointer;
uint8_t pByte;
if (totalAlpha == 0xFF)
{
do
{
// Use alpha from covers directly
const uint8_t alpha = *covers++;
if (alpha == 0xFF)
{
// Solid pixel
*p++ = *src++;
*p++ = *src++;
*p++ = *src++;
}
else
{
const uint8_t ialpha = 0xFF - alpha;
pByte = *p;
*p++ = LCD::div255(*src++ * alpha + pByte * ialpha);
pByte = *p;
*p++ = LCD::div255(*src++ * alpha + pByte * ialpha);
pByte = *p;
*p++ = LCD::div255(*src++ * alpha + pByte * ialpha);
}
} while (--count != 0);
}
else
{
do
{
const uint8_t alpha = LCD::div255((*covers++) * totalAlpha);
const uint8_t ialpha = 0xFF - alpha;
pByte = *p;
*p++ = LCD::div255(*src++ * alpha + pByte * ialpha);
pByte = *p;
*p++ = LCD::div255(*src++ * alpha + pByte * ialpha);
pByte = *p;
*p++ = LCD::div255(*src++ * alpha + pByte * ialpha);
} while (--count != 0);
}
}
else if (bitmapARGB8888Pointer)
{
const uint32_t* src = bitmapARGB8888Pointer;
if (totalAlpha == 0xFF)
{
do
{
const uint8_t srcAlpha = (*src) >> 24;
const uint8_t alpha = LCD::div255((*covers++) * srcAlpha);
if (alpha == 0xFF)
{
// Solid pixel
*p++ = (*src); // Blue
*p++ = (*src) >> 8; // Green
*p++ = (*src) >> 16; // Red
}
else
{
// Non-Transparent pixel
const uint8_t ialpha = 0xFF - alpha;
uint8_t pByte = *p;
uint8_t cByte = (*src);
*p++ = LCD::div255(cByte * alpha + pByte * ialpha);
pByte = *p;
cByte = (*src) >> 8;
*p++ = LCD::div255(cByte * alpha + pByte * ialpha);
pByte = *p;
cByte = (*src) >> 16;
*p++ = LCD::div255(cByte * alpha + pByte * ialpha);
}
src++;
} while (--count != 0);
}
else
{
do
{
const uint8_t srcAlpha = (*src) >> 24;
const uint8_t alpha = LCD::div255((*covers++) * LCD::div255(srcAlpha * totalAlpha));
if (alpha)
{
const uint8_t ialpha = 0xFF - alpha;
uint8_t pByte = *p;
uint8_t cByte = (*src);
*p++ = LCD::div255(cByte * alpha + pByte * ialpha);
pByte = *p;
cByte = (*src) >> 8;
*p++ = LCD::div255(cByte * alpha + pByte * ialpha);
pByte = *p;
cByte = (*src) >> 16;
*p++ = LCD::div255(cByte * alpha + pByte * ialpha);
}
else
{
p += 3;
}
src++;
} while (--count != 0);
}
}
}
bool PainterRGB888Bitmap::renderInit()
{
bitmapARGB8888Pointer = 0;
bitmapRGB888Pointer = 0;
if (bitmap.getId() == BITMAP_INVALID)
{
return false;
}
if ((currentX >= bitmapRectToFrameBuffer.width) || (currentY >= bitmapRectToFrameBuffer.height))
{
// Outside bitmap area, do not draw anything
// Consider the following instead of "return" to get a tiled image:
// currentX %= bitmapRectToFrameBuffer.width
// currentY %= bitmapRectToFrameBuffer.height
return false;
}
if (bitmap.getFormat() == Bitmap::ARGB8888)
{
bitmapARGB8888Pointer = (const uint32_t*)bitmap.getData();
if (!bitmapARGB8888Pointer)
{
return false;
}
bitmapARGB8888Pointer += currentX + currentY * bitmapRectToFrameBuffer.width;
return true;
}
if (bitmap.getFormat() == Bitmap::RGB888)
{
bitmapRGB888Pointer = bitmap.getData();
if (!bitmapRGB888Pointer)
{
return false;
}
bitmapRGB888Pointer += (currentX + currentY * bitmapRectToFrameBuffer.width) * 3;
return true;
}
return false;
}
bool PainterRGB888Bitmap::renderNext(uint8_t& red, uint8_t& green, uint8_t& blue, uint8_t& alpha)
{
if (currentX >= bitmapRectToFrameBuffer.width)
{
return false;
}
if (bitmapARGB8888Pointer != 0)
{
uint32_t argb8888 = *bitmapARGB8888Pointer++;
alpha = (argb8888 >> 24) & 0xFF;
red = (argb8888 >> 16) & 0xFF;
green = (argb8888 >> 8) & 0xFF;
blue = (argb8888) & 0xFF;
}
else if (bitmapRGB888Pointer != 0)
{
blue = *bitmapRGB888Pointer++;
green = *bitmapRGB888Pointer++;
red = *bitmapRGB888Pointer++;
alpha = 0xFF; // No alpha per pixel in the image, it is solid
}
// Apply given alpha from setAlpha()
alpha = LCD::div255(alpha * painterAlpha);
return true;
}
} // namespace touchgfx

View File

@ -0,0 +1,209 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <touchgfx/widgets/canvas/PainterRGB888L8Bitmap.hpp>
namespace touchgfx
{
void PainterRGB888L8Bitmap::setBitmap(const Bitmap& bmp)
{
bitmap = bmp;
assert((bitmap.getId() == BITMAP_INVALID || bitmap.getFormat() == Bitmap::L8) && "The chosen painter only works with appropriate L8 bitmaps");
bitmapRectToFrameBuffer = bitmap.getRect();
DisplayTransformation::transformDisplayToFrameBuffer(bitmapRectToFrameBuffer);
}
void PainterRGB888L8Bitmap::render(uint8_t* ptr, int x, int xAdjust, int y, unsigned count, const uint8_t* covers)
{
uint8_t* p = ptr + ((x + xAdjust) * 3);
currentX = x + areaOffsetX;
currentY = y + areaOffsetY;
if (!renderInit())
{
return;
}
if (currentX + (int)count > bitmapRectToFrameBuffer.width)
{
count = bitmapRectToFrameBuffer.width - currentX;
}
const uint8_t totalAlpha = LCD::div255(widgetAlpha * painterAlpha);
if ((Bitmap::ClutFormat)((const uint16_t*)bitmapExtraPointer)[-2] == Bitmap::CLUT_FORMAT_L8_RGB888)
{
if (totalAlpha == 0xFF)
{
do
{
const uint8_t* src = &bitmapExtraPointer[*bitmapPointer++ * 3];
// Use alpha from covers directly
const uint8_t alpha = *covers++;
if (alpha == 0xFF)
{
// Solid pixel
*p++ = *src++;
*p++ = *src++;
*p++ = *src;
}
else
{
const uint8_t ialpha = 0xFF - alpha;
uint8_t pByte = *p;
*p++ = LCD::div255(*src++ * alpha + pByte * ialpha);
pByte = *p;
*p++ = LCD::div255(*src++ * alpha + pByte * ialpha);
pByte = *p;
*p++ = LCD::div255(*src * alpha + pByte * ialpha);
}
} while (--count != 0);
}
else
{
do
{
const uint8_t* src = &bitmapExtraPointer[*bitmapPointer++ * 3];
const uint8_t alpha = LCD::div255((*covers++) * totalAlpha);
const uint8_t ialpha = 0xFF - alpha;
uint8_t pByte = *p;
*p++ = LCD::div255(*src++ * alpha + pByte * ialpha);
pByte = *p;
*p++ = LCD::div255(*src++ * alpha + pByte * ialpha);
pByte = *p;
*p++ = LCD::div255(*src * alpha + pByte * ialpha);
} while (--count != 0);
}
}
else // Bitmap::CLUT_FORMAT_L8_ARGB8888
{
if (totalAlpha == 0xFF)
{
do
{
uint32_t src = ((const uint32_t*)bitmapExtraPointer)[*bitmapPointer++];
const uint8_t srcAlpha = src >> 24;
const uint8_t alpha = LCD::div255((*covers++) * srcAlpha);
if (alpha == 0xFF)
{
// Solid pixel
*p++ = src; // Blue
*p++ = src >> 8; // Green
*p++ = src >> 16; // Red
}
else
{
// Non-Transparent pixel
const uint8_t ialpha = 0xFF - alpha;
uint8_t pByte = *p;
*p++ = LCD::div255((src & 0xFF) * alpha + pByte * ialpha);
pByte = *p;
*p++ = LCD::div255(((src >> 8) & 0xFF) * alpha + pByte * ialpha);
pByte = *p;
*p++ = LCD::div255(((src >> 16) & 0xFF) * alpha + pByte * ialpha);
}
} while (--count != 0);
}
else
{
do
{
uint32_t src = ((const uint32_t*)bitmapExtraPointer)[*bitmapPointer++];
const uint8_t srcAlpha = src >> 24;
const uint8_t alpha = LCD::div255((*covers++) * LCD::div255(srcAlpha * totalAlpha));
if (alpha)
{
const uint8_t ialpha = 0xFF - alpha;
uint8_t pByte = *p;
*p++ = LCD::div255((src & 0xFF) * alpha + pByte * ialpha);
pByte = *p;
*p++ = LCD::div255(((src >> 8) & 0xFF) * alpha + pByte * ialpha);
pByte = *p;
*p++ = LCD::div255(((src >> 16) & 0xFF) * alpha + pByte * ialpha);
}
else
{
p += 3;
}
} while (--count != 0);
}
}
}
bool PainterRGB888L8Bitmap::renderInit()
{
bitmapPointer = 0;
bitmapExtraPointer = 0;
if (bitmap.getId() == BITMAP_INVALID)
{
return false;
}
if ((currentX >= bitmapRectToFrameBuffer.width) || (currentY >= bitmapRectToFrameBuffer.height))
{
// Outside bitmap area, do not draw anything
// Consider the following instead of "return" to get a tiled image:
// currentX %= bitmapRectToFrameBuffer.width
// currentY %= bitmapRectToFrameBuffer.height
return false;
}
if (bitmap.getFormat() == Bitmap::L8)
{
bitmapPointer = bitmap.getData();
if (!bitmapPointer)
{
return false;
}
bitmapPointer += currentX + currentY * bitmapRectToFrameBuffer.width;
bitmapExtraPointer = bitmap.getExtraData();
assert((bitmapExtraPointer != 0 && (*(const uint16_t*)bitmapExtraPointer == Bitmap::CLUT_FORMAT_L8_RGB888 || *(const uint16_t*)bitmapExtraPointer == Bitmap::CLUT_FORMAT_L8_ARGB8888)));
bitmapExtraPointer += 4; // Skip header
return true;
}
return false;
}
bool PainterRGB888L8Bitmap::renderNext(uint8_t& red, uint8_t& green, uint8_t& blue, uint8_t& alpha)
{
if (currentX >= bitmapRectToFrameBuffer.width)
{
return false;
}
if ((Bitmap::ClutFormat)((const uint16_t*)bitmapExtraPointer)[-2] == Bitmap::CLUT_FORMAT_L8_RGB888)
{
const uint8_t* clut = bitmapExtraPointer + (*bitmapPointer++) * 3;
blue = *clut++;
green = *clut++;
red = *clut;
alpha = 0xFF; // No alpha per pixel in the image, it is solid
}
else // L8_ARGB8888
{
const uint32_t* clut = ((const uint32_t*)bitmapExtraPointer) + (*bitmapPointer++);
uint32_t argb8888 = *(const uint32_t*)clut;
alpha = (argb8888 >> 24) & 0xFF;
red = (argb8888 >> 16) & 0xFF;
green = (argb8888 >> 8) & 0xFF;
blue = (argb8888) & 0xFF;
}
// Apply given alpha from setAlpha()
alpha = LCD::div255(alpha * painterAlpha);
return true;
}
} // namespace touchgfx

View File

@ -0,0 +1,68 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <platform/driver/lcd/LCD8bpp_RGBA2222.hpp>
#include <touchgfx/Color.hpp>
#include <touchgfx/widgets/canvas/PainterRGBA2222.hpp>
namespace touchgfx
{
void PainterRGBA2222::render(uint8_t* ptr, int x, int xAdjust, int /*y*/, unsigned count, const uint8_t* covers)
{
uint8_t* p = ptr + (x + xAdjust);
const uint8_t totalAlpha = LCD::div255(widgetAlpha * painterAlpha);
if (totalAlpha == 0xFF)
{
do
{
const uint8_t alpha = *covers++;
if (alpha == 0xFF)
{
*p = painterColor;
}
else
{
*p = mixColors(painterRed, painterGreen, painterBlue, *p, alpha);
}
p++;
} while (--count != 0);
}
else
{
do
{
const uint8_t alpha = LCD::div255((*covers++) * totalAlpha);
if (alpha == 0xFF)
{
*p = painterColor;
}
else
{
*p = mixColors(painterRed, painterGreen, painterBlue, *p, alpha);
}
p++;
} while (--count != 0);
}
}
bool PainterRGBA2222::renderNext(uint8_t& red, uint8_t& green, uint8_t& blue, uint8_t& alpha)
{
red = painterRed;
green = painterGreen;
blue = painterBlue;
alpha = painterAlpha;
return true;
}
} // namespace touchgfx

View File

@ -0,0 +1,139 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <touchgfx/widgets/canvas/PainterRGBA2222Bitmap.hpp>
namespace touchgfx
{
void PainterRGBA2222Bitmap::setBitmap(const Bitmap& bmp)
{
bitmap = bmp;
assert((bitmap.getId() == BITMAP_INVALID || bitmap.getFormat() == Bitmap::RGBA2222) && "The chosen painter only works with RGBA2222 bitmaps");
bitmapRectToFrameBuffer = bitmap.getRect();
DisplayTransformation::transformDisplayToFrameBuffer(bitmapRectToFrameBuffer);
}
void PainterRGBA2222Bitmap::render(uint8_t* ptr,
int x,
int xAdjust,
int y,
unsigned count,
const uint8_t* covers)
{
uint8_t* p = ptr + (x + xAdjust);
currentX = x + areaOffsetX;
currentY = y + areaOffsetY;
if (!renderInit())
{
return;
}
if (currentX + (int)count > bitmapRectToFrameBuffer.width)
{
count = bitmapRectToFrameBuffer.width - currentX;
}
const uint8_t totalAlpha = LCD::div255(widgetAlpha * painterAlpha);
const uint8_t* src = bitmapRGBA2222Pointer;
if (totalAlpha == 0xFF)
{
do
{
const uint8_t srcAlpha = ((*src) & 0x03) * 0x55;
const uint8_t alpha = LCD::div255((*covers++) * srcAlpha);
if (alpha == 0xFF)
{
// Solid pixel
*p = *src;
}
else if (alpha)
{
// Non-Transparent pixel
*p = mixColors(*src, *p, alpha);
}
p++;
src++;
} while (--count != 0);
}
else
{
do
{
const uint8_t srcAlpha = ((*src) & 0x03) * 0x55;
const uint8_t alpha = LCD::div255((*covers++) * srcAlpha);
if (alpha) // This can never get to max=0xFF*0xFF as totalAlpha<255
{
// Non-Transparent pixel
*p = mixColors(*src, *p, alpha);
}
p++;
src++;
} while (--count != 0);
}
}
bool PainterRGBA2222Bitmap::renderInit()
{
bitmapRGBA2222Pointer = 0;
if (bitmap.getId() == BITMAP_INVALID)
{
return false;
}
if ((currentX >= bitmapRectToFrameBuffer.width) || (currentY >= bitmapRectToFrameBuffer.height))
{
// Outside bitmap area, do not draw anything
// Consider the following instead of "return" to get a tiled image:
// currentX %= bitmapRectToFrameBuffer.width
// currentY %= bitmapRectToFrameBuffer.height
return false;
}
if (bitmap.getFormat() == Bitmap::RGBA2222)
{
bitmapRGBA2222Pointer = bitmap.getData();
if (!bitmapRGBA2222Pointer)
{
return false;
}
bitmapRGBA2222Pointer += currentX + currentY * bitmapRectToFrameBuffer.width;
return true;
}
return false;
}
bool PainterRGBA2222Bitmap::renderNext(uint8_t& red, uint8_t& green, uint8_t& blue, uint8_t& alpha)
{
if (currentX >= bitmapRectToFrameBuffer.width)
{
return false;
}
else if (bitmapRGBA2222Pointer != 0)
{
uint16_t RGBA2222 = *bitmapRGBA2222Pointer++;
red = LCD8bpp_RGBA2222::getRedFromColor(RGBA2222);
green = LCD8bpp_RGBA2222::getGreenFromColor(RGBA2222);
blue = LCD8bpp_RGBA2222::getBlueFromColor(RGBA2222);
alpha = (RGBA2222 & 0x03) * 0x55; // To get full range 0-0xFF
}
// Apply given alpha from setAlpha()
alpha = LCD::div255(alpha * painterAlpha);
return true;
}
} // namespace touchgfx

View File

@ -0,0 +1,757 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <touchgfx/widgets/graph/AbstractDataGraph.hpp>
#include <touchgfx/widgets/graph/GraphElements.hpp>
#include <touchgfx/widgets/graph/GraphLabels.hpp>
namespace touchgfx
{
AbstractDataGraph::AbstractDataGraph(int16_t capacity)
: dataScale(1), alpha(255),
graphArea(), leftArea(), rightArea(), topArea(), bottomArea(),
topPadding(0), leftPadding(0), rightPadding(0), bottomPadding(0),
maxCapacity(capacity), usedCapacity(0), gapBeforeIndex(0), clickAction(0), dragAction(0)
{
add(graphArea);
add(topArea);
add(leftArea);
add(rightArea);
add(bottomArea);
// Place all areas properly:
setGraphAreaMargin(0, 0, 0, 0);
setTouchable(true);
}
void AbstractDataGraph::setScale(int scale)
{
dataScale = scale;
}
int AbstractDataGraph::getScale() const
{
return dataScale;
}
void AbstractDataGraph::setAlpha(uint8_t newAlpha)
{
alpha = newAlpha;
}
uint8_t AbstractDataGraph::getAlpha() const
{
return alpha;
}
void AbstractDataGraph::setWidth(int16_t width)
{
topArea.setWidth(width);
bottomArea.setWidth(width);
rightArea.setX(width - rightArea.getWidth());
graphArea.setWidth(width - (leftArea.getWidth() + rightArea.getWidth()));
Container::setWidth(width);
updateAreasPosition();
}
void AbstractDataGraph::setHeight(int16_t height)
{
leftArea.setHeight(height);
rightArea.setHeight(height);
bottomArea.setY(height - bottomArea.getHeight());
graphArea.setHeight(height - (topArea.getHeight() + bottomArea.getHeight()));
Container::setHeight(height);
updateAreasPosition();
}
void AbstractDataGraph::setGraphAreaMargin(int16_t top, int16_t left, int16_t right, int16_t bottom)
{
graphArea.setPosition(left, top, getWidth() - (left + right), getHeight() - (top + bottom));
topArea.setPosition(0, 0, getWidth(), top);
leftArea.setPosition(0, 0, left, getHeight());
rightArea.setPosition(getWidth() - right, 0, right, getHeight());
bottomArea.setPosition(0, getHeight() - bottom, getWidth(), bottom);
updateAreasPosition();
}
int16_t AbstractDataGraph::getGraphAreaMarginTop() const
{
return topArea.getHeight();
}
int16_t AbstractDataGraph::getGraphAreaMarginLeft() const
{
return leftArea.getWidth();
}
int16_t AbstractDataGraph::getGraphAreaMarginRight() const
{
return rightArea.getWidth();
}
int16_t AbstractDataGraph::getGraphAreaMarginBottom() const
{
return bottomArea.getHeight();
}
void AbstractDataGraph::setGraphAreaPadding(int16_t top, int16_t left, int16_t right, int16_t bottom)
{
topPadding = top;
leftPadding = left;
rightPadding = right;
bottomPadding = bottom;
}
int16_t AbstractDataGraph::getGraphAreaPaddingTop() const
{
return topPadding;
}
int16_t AbstractDataGraph::getGraphAreaPaddingLeft() const
{
return leftPadding;
}
int16_t AbstractDataGraph::getGraphAreaPaddingRight() const
{
return rightPadding;
}
int16_t AbstractDataGraph::getGraphAreaPaddingBottom() const
{
return bottomPadding;
}
int16_t AbstractDataGraph::getGraphAreaWidth() const
{
return graphArea.getWidth() - (leftPadding + rightPadding);
}
int16_t AbstractDataGraph::getGraphAreaWidthIncludingPadding() const
{
return graphArea.getWidth();
}
int16_t AbstractDataGraph::getGraphAreaHeight() const
{
return graphArea.getHeight() - (topPadding + bottomPadding);
}
int16_t AbstractDataGraph::getGraphAreaHeightIncludingPadding() const
{
return graphArea.getHeight();
}
void AbstractDataGraph::setGraphRange(int xMin, int xMax, int yMin, int yMax)
{
setGraphRangeX(xMin, xMax);
setGraphRangeY(yMin, yMax);
}
void AbstractDataGraph::setGraphRange(int xMin, int xMax, float yMin, float yMax)
{
setGraphRangeX(xMin, xMax);
setGraphRangeY(yMin, yMax);
}
void AbstractDataGraph::clear()
{
usedCapacity = 0;
}
void AbstractDataGraph::setGapBeforeIndex(int16_t index)
{
gapBeforeIndex = index;
}
int16_t AbstractDataGraph::getGapBeforeIndex() const
{
return gapBeforeIndex;
}
void AbstractDataGraph::addGraphElement(AbstractGraphElement& d)
{
graphArea.add(d);
d.setPosition(0, 0, graphArea.getWidth(), graphArea.getHeight());
}
void AbstractDataGraph::addTopElement(AbstractGraphDecoration& d)
{
topArea.add(d);
d.setPosition(0, 0, topArea.getWidth(), topArea.getHeight());
}
void AbstractDataGraph::addLeftElement(AbstractGraphDecoration& d)
{
leftArea.add(d);
d.setPosition(0, 0, leftArea.getWidth(), leftArea.getHeight());
}
void AbstractDataGraph::addRightElement(AbstractGraphDecoration& d)
{
rightArea.add(d);
d.setPosition(0, 0, rightArea.getWidth(), rightArea.getHeight());
}
void AbstractDataGraph::addBottomElement(AbstractGraphDecoration& d)
{
bottomArea.add(d);
d.setPosition(0, 0, bottomArea.getWidth(), bottomArea.getHeight());
}
int16_t AbstractDataGraph::getUsedCapacity() const
{
return usedCapacity;
}
int16_t AbstractDataGraph::getMaxCapacity() const
{
return maxCapacity;
}
bool AbstractDataGraph::getNearestIndexForScreenXY(int16_t x, int16_t y, int16_t& index)
{
if (usedCapacity == 0)
{
return false;
}
int32_t bestDist = 0x7FFFFFFF;
for (int16_t ix = 0; ix < usedCapacity; ix++)
{
const int16_t xDist = abs(indexToScreenX(ix) - x);
const int16_t yDist = abs(indexToScreenY(ix) - y);
const int32_t dist = xDist * xDist + yDist * yDist;
if (dist < bestDist)
{
index = ix;
bestDist = dist;
}
}
return true;
}
bool AbstractDataGraph::getNearestIndexForScreenX(int16_t x, int16_t& index) const
{
if (usedCapacity == 0)
{
return false;
}
int16_t indexLow;
int16_t indexHigh;
xScreenRangeToIndexRange(x, x, indexLow, indexHigh);
const int16_t xLow = indexToScreenX(indexLow);
const int16_t xHigh = indexToScreenX(indexHigh);
index = abs(xLow - x) <= abs(xHigh - x) ? indexLow : indexHigh;
return true;
}
int16_t AbstractDataGraph::indexToScreenX(int16_t index) const
{
return indexToScreenXQ5(index).round();
}
int16_t AbstractDataGraph::indexToScreenY(int16_t index) const
{
return indexToScreenYQ5(index).round();
}
int AbstractDataGraph::indexToDataPointXAsInt(int16_t index) const
{
return scaled2int(indexToDataPointXScaled(index));
}
float AbstractDataGraph::indexToDataPointXAsFloat(int16_t index) const
{
return scaled2float(indexToDataPointXScaled(index));
}
int AbstractDataGraph::indexToDataPointYAsInt(int16_t index) const
{
return scaled2int(indexToDataPointYScaled(index));
}
float AbstractDataGraph::indexToDataPointYAsFloat(int16_t index) const
{
return scaled2float(indexToDataPointYScaled(index));
}
int32_t AbstractDataGraph::indexToGlobalIndex(int16_t index) const
{
return (int32_t)index;
}
void AbstractDataGraph::setClickAction(GenericCallback<const AbstractDataGraph&, const GraphClickEvent&>& callback)
{
clickAction = &callback;
}
void AbstractDataGraph::setDragAction(GenericCallback<const AbstractDataGraph&, const GraphDragEvent&>& callback)
{
dragAction = &callback;
}
void AbstractDataGraph::handleClickEvent(const ClickEvent& event)
{
if (event.getType() == ClickEvent::CANCEL)
{
return;
}
const int16_t x = event.getX() - graphArea.getX();
int16_t index;
if (getNearestIndexForScreenX(x, index))
{
if (event.getType() == ClickEvent::PRESSED || event.getType() == ClickEvent::RELEASED)
{
if (clickAction && clickAction->isValid())
{
GraphClickEvent graphClickEvent(index, event);
clickAction->execute(*this, graphClickEvent);
}
}
}
}
void AbstractDataGraph::handleDragEvent(const DragEvent& event)
{
const int16_t x = event.getNewX() - graphArea.getX();
int16_t index;
if (getNearestIndexForScreenX(x, index))
{
if (event.getType() == DragEvent::DRAGGED)
{
if (dragAction && dragAction->isValid())
{
GraphDragEvent graphDragEvent(index, event);
dragAction->execute(*this, graphDragEvent);
}
}
}
}
void AbstractDataGraph::invalidateGraphPointAt(int16_t index)
{
if (index < usedCapacity)
{
AbstractGraphElement* d = (AbstractGraphElement*)graphArea.getFirstChild();
while (d)
{
d->invalidateGraphPointAt(index);
d = (AbstractGraphElement*)d->getNextSibling();
}
}
}
void AbstractDataGraph::invalidateGraphArea()
{
graphArea.invalidate();
}
void AbstractDataGraph::invalidateXAxisPointAt(int16_t index)
{
AbstractGraphElement* d = (AbstractGraphElement*)topArea.getFirstChild();
while (d)
{
d->invalidateGraphPointAt(index);
d = (AbstractGraphElement*)d->getNextSibling();
}
d = (AbstractGraphElement*)bottomArea.getFirstChild();
while (d)
{
d->invalidateGraphPointAt(index);
d = (AbstractGraphElement*)d->getNextSibling();
}
}
void AbstractDataGraph::invalidateAllXAxisPoints()
{
int min = getGraphRangeXMin();
int max = getGraphRangeXMax();
if (max < min)
{
const int16_t tmp = min;
min = max;
max = tmp;
}
for (int index = min; index <= max; index++)
{
invalidateXAxisPointAt(index);
}
}
void AbstractDataGraph::updateAreasPosition()
{
Drawable* d = graphArea.getFirstChild();
while (d)
{
d->setPosition(0, 0, graphArea.getWidth(), graphArea.getHeight());
d = d->getNextSibling();
}
d = topArea.getFirstChild();
while (d)
{
d->setPosition(0, 0, topArea.getWidth(), topArea.getHeight());
d = d->getNextSibling();
}
d = leftArea.getFirstChild();
while (d)
{
d->setPosition(0, 0, leftArea.getWidth(), leftArea.getHeight());
d = d->getNextSibling();
}
d = rightArea.getFirstChild();
while (d)
{
d->setPosition(0, 0, rightArea.getWidth(), rightArea.getHeight());
d = d->getNextSibling();
}
d = bottomArea.getFirstChild();
while (d)
{
d->setPosition(0, 0, bottomArea.getWidth(), bottomArea.getHeight());
d = d->getNextSibling();
}
invalidate();
}
void AbstractDataGraph::setGraphRangeScaled(int xMin, int xMax, int yMin, int yMax)
{
setGraphRangeX(xMin, xMax);
setGraphRangeYScaled(yMin, yMax);
}
int AbstractDataGraph::convertToGraphScale(int value, int scale) const
{
if (scale == dataScale)
{
return value;
}
return muldiv(value, dataScale, scale);
}
int AbstractDataGraph::getXAxisScaleScaled() const
{
return dataScale;
}
int AbstractDataGraph::getXAxisOffsetScaled() const
{
return 0;
}
AbstractDataGraphWithY::AbstractDataGraphWithY(int16_t capacity, int* values)
: AbstractDataGraph(capacity), yValues(values), dataCounter(0), xOffset(0), xScale(1),
graphRangeMinX(0), graphRangeMaxX(0), graphRangeMinY(0), graphRangeMaxY(0)
{
assert(capacity > 0);
setGraphRangeX(0, capacity - 1);
}
int16_t AbstractDataGraphWithY::addDataPoint(int y)
{
return addDataPointScaled(y * dataScale);
}
int16_t AbstractDataGraphWithY::addDataPoint(float y)
{
return addDataPointScaled(float2scaled(y));
}
void AbstractDataGraphWithY::setGraphRangeX(int min, int max)
{
assert(min != max);
if (max < min)
{
const int tmp = min;
min = max;
max = tmp;
}
if (min != graphRangeMinX || max != graphRangeMaxX)
{
graphRangeMinX = min;
graphRangeMaxX = max;
topArea.invalidate();
graphArea.invalidate();
bottomArea.invalidate();
}
}
void AbstractDataGraphWithY::setGraphRangeY(int min, int max)
{
setGraphRangeYScaled(int2scaled(min), int2scaled(max));
}
void AbstractDataGraphWithY::setGraphRangeY(float min, float max)
{
setGraphRangeYScaled(float2scaled(min), float2scaled(max));
}
int AbstractDataGraphWithY::getGraphRangeXMin() const
{
return graphRangeMinX;
}
int AbstractDataGraphWithY::getGraphRangeXMax() const
{
return graphRangeMaxX;
}
int AbstractDataGraphWithY::getGraphRangeYMinAsInt() const
{
return scaled2int(graphRangeMinY);
}
float AbstractDataGraphWithY::getGraphRangeYMinAsFloat() const
{
return scaled2float(graphRangeMinY);
}
int AbstractDataGraphWithY::getGraphRangeYMaxAsInt() const
{
return scaled2int(graphRangeMaxY);
}
float AbstractDataGraphWithY::getGraphRangeYMaxAsFloat() const
{
return scaled2float(graphRangeMaxY);
}
void AbstractDataGraphWithY::setGraphRangeYAuto(bool showXaxis, int margin)
{
if (usedCapacity > 0)
{
int loX = getGraphRangeXMin();
int hiX = getGraphRangeXMax();
if (loX > hiX)
{
const int16_t tmp = loX;
loX = hiX;
hiX = tmp;
}
loX = MAX(loX, 0);
hiX = MIN(hiX, usedCapacity);
if (loX < usedCapacity && hiX >= 0)
{
int loYnew = showXaxis ? margin : yValues[loX];
int hiYnew = showXaxis ? -margin : yValues[loX];
for (int16_t i = loX; i < hiX; i++)
{
int y = yValues[i];
if (loYnew > y)
{
loYnew = y;
}
if (hiYnew < y)
{
hiYnew = y;
}
}
loYnew -= margin;
hiYnew += margin;
if (loYnew != hiYnew)
{
setGraphRangeYScaled(loYnew, hiYnew);
}
}
}
}
void AbstractDataGraphWithY::setScale(int scale)
{
const int oldScale = getScale();
AbstractDataGraph::setScale(scale);
xScale = convertToGraphScale(xScale, oldScale);
xOffset = convertToGraphScale(xOffset, oldScale);
}
void AbstractDataGraphWithY::setXAxisScale(int scale)
{
setXAxisScaleScaled(scale * dataScale);
}
void AbstractDataGraphWithY::setXAxisScale(float scale)
{
setXAxisScaleScaled(float2scaled(scale));
}
int AbstractDataGraphWithY::getXAxisScaleAsInt() const
{
return scaled2int(getXAxisScaleScaled());
}
float AbstractDataGraphWithY::getXAxisScaleAsFloat() const
{
return scaled2float(getXAxisScaleScaled());
}
void AbstractDataGraphWithY::setXAxisOffset(int offset)
{
setXAxisOffsetScaled(offset * dataScale);
}
void AbstractDataGraphWithY::setXAxisOffset(float offset)
{
setXAxisOffsetScaled(float2scaled(offset));
}
int AbstractDataGraphWithY::getXAxisOffsetAsInt() const
{
return scaled2int(getXAxisOffsetScaled());
}
float AbstractDataGraphWithY::getXAxisOffsetAsFloat() const
{
return scaled2float(getXAxisOffsetScaled());
}
int16_t AbstractDataGraphWithY::addDataPointScaled(int y)
{
beforeAddValue();
dataCounter++;
return addValue(y);
}
void AbstractDataGraphWithY::beforeAddValue()
{
}
int16_t AbstractDataGraphWithY::realIndex(int16_t index) const
{
return index;
}
int AbstractDataGraphWithY::indexToDataPointXScaled(int16_t index) const
{
return (indexToGlobalIndex(index) * getXAxisScaleScaled()) + getXAxisOffsetScaled();
}
int AbstractDataGraphWithY::indexToDataPointYScaled(int16_t index) const
{
return yValues[realIndex(index)];
}
bool AbstractDataGraphWithY::xScreenRangeToIndexRange(int16_t xLo, int16_t xHi, int16_t& indexLow, int16_t& indexHigh) const
{
if (usedCapacity == 0)
{
indexLow = indexHigh = -1;
return false;
}
if (getGraphAreaWidth() == 1)
{
indexLow = 0;
indexHigh = usedCapacity - 1;
return true;
}
CWRUtil::Q5 loQ5 = CWRUtil::muldivQ5(CWRUtil::toQ5(xLo - leftPadding), CWRUtil::Q5(graphRangeMaxX - graphRangeMinX), CWRUtil::Q5(getGraphAreaWidth() - 1)) + CWRUtil::toQ5(graphRangeMinX);
CWRUtil::Q5 hiQ5 = CWRUtil::muldivQ5(CWRUtil::toQ5(xHi - leftPadding), CWRUtil::Q5(graphRangeMaxX - graphRangeMinX), CWRUtil::Q5(getGraphAreaWidth() - 1)) + CWRUtil::toQ5(graphRangeMinX);
if (loQ5 > hiQ5)
{
const CWRUtil::Q5 tmp = loQ5;
loQ5 = hiQ5;
hiQ5 = tmp;
}
const int low = loQ5.to<int>();
const int high = hiQ5.ceil();
const int lowValid = 0;
const int highValid = usedCapacity - 1;
if (high < lowValid)
{
indexLow = indexHigh = lowValid;
return false;
}
if (low > highValid)
{
indexLow = indexHigh = highValid;
return false;
}
indexLow = MAX(lowValid, low);
indexHigh = MIN(highValid, high);
return true;
}
void AbstractDataGraphWithY::setXAxisScaleScaled(int scale)
{
xScale = scale;
}
int AbstractDataGraphWithY::getXAxisScaleScaled() const
{
return xScale;
}
void AbstractDataGraphWithY::setGraphRangeYScaled(int min, int max)
{
assert(min != max);
if (max < min)
{
const int tmp = min;
min = max;
max = tmp;
}
if (min != graphRangeMinY || max != graphRangeMaxY)
{
graphRangeMinY = min;
graphRangeMaxY = max;
leftArea.invalidate();
graphArea.invalidate();
rightArea.invalidate();
}
}
int AbstractDataGraphWithY::getGraphRangeYMinScaled() const
{
return graphRangeMinY;
}
int AbstractDataGraphWithY::getGraphRangeYMaxScaled() const
{
return graphRangeMaxY;
}
void AbstractDataGraphWithY::setXAxisOffsetScaled(int offset)
{
xOffset = offset;
}
int AbstractDataGraphWithY::getXAxisOffsetScaled() const
{
return xOffset;
}
CWRUtil::Q5 AbstractDataGraphWithY::valueToScreenXQ5(int x) const
{
return CWRUtil::muldiv_toQ5(x - graphRangeMinX, getGraphAreaWidth() - 1, graphRangeMaxX - graphRangeMinX) + CWRUtil::toQ5(leftPadding);
}
CWRUtil::Q5 AbstractDataGraphWithY::valueToScreenYQ5(int y) const
{
return CWRUtil::toQ5(getGraphAreaHeight() + topPadding - 1) - CWRUtil::muldiv_toQ5(y - graphRangeMinY, getGraphAreaHeight() - 1, graphRangeMaxY - graphRangeMinY);
}
CWRUtil::Q5 AbstractDataGraphWithY::indexToScreenXQ5(int16_t index) const
{
return valueToScreenXQ5(index);
}
CWRUtil::Q5 AbstractDataGraphWithY::indexToScreenYQ5(int16_t index) const
{
return valueToScreenYQ5(yValues[realIndex(index)]);
}
} // namespace touchgfx

View File

@ -0,0 +1,551 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <touchgfx/widgets/graph/GraphLabels.hpp>
namespace touchgfx
{
GraphLabelsBase::GraphLabelsBase()
: labelInterval(0), labelTypedText(TypedText(TYPED_TEXT_INVALID)), labelRotation(TEXT_ROTATE_0), labelDecimals(0), labelDecimalPoint('.'),
majorLabel(0)
{
}
int GraphLabelsBase::getCorrectlyScaledMajorInterval(const AbstractDataGraph* graph) const
{
return majorLabel == 0 ? 0 : convertToGraphScale(graph, majorLabel->getIntervalScaled(), majorLabel->getScale());
}
void GraphLabelsBase::setIntervalScaled(int interval)
{
labelInterval = abs(interval);
}
void GraphLabelsBase::setInterval(int interval)
{
setIntervalScaled(abs(interval) * dataScale);
}
void GraphLabelsBase::setInterval(float interval)
{
setIntervalScaled(AbstractDataGraph::float2scaled(abs(interval), dataScale));
}
int GraphLabelsBase::getIntervalScaled() const
{
return labelInterval;
}
int GraphLabelsBase::getIntervalAsInt() const
{
return AbstractDataGraph::scaled2int(getIntervalScaled(), dataScale);
}
float GraphLabelsBase::getIntervalAsFloat() const
{
return AbstractDataGraph::scaled2float(getIntervalScaled(), dataScale);
}
void GraphLabelsBase::setMajorLabel(const GraphLabelsBase& major)
{
majorLabel = &major;
}
void GraphLabelsBase::setLabelTypedText(const TypedText& typedText)
{
labelTypedText = typedText;
}
TypedText GraphLabelsBase::getLabelTypedText() const
{
return labelTypedText;
}
void GraphLabelsBase::setLabelRotation(TextRotation rotation)
{
labelRotation = rotation;
}
TextRotation GraphLabelsBase::getLabelRotation() const
{
return labelRotation;
}
void GraphLabelsBase::setLabelDecimals(uint16_t decimals)
{
labelDecimals = decimals;
}
uint16_t GraphLabelsBase::getLabelDecimals() const
{
return labelDecimals;
}
void GraphLabelsBase::setLabelDecimalPoint(Unicode::UnicodeChar decimalPoint)
{
labelDecimalPoint = decimalPoint;
}
Unicode::UnicodeChar GraphLabelsBase::getLabelDecimalPoint() const
{
return labelDecimalPoint;
}
void GraphLabelsBase::invalidateGraphPointAt(int16_t index)
{
}
void GraphLabelsBase::formatLabel(Unicode::UnicodeChar* buffer, int16_t bufferSize, int label, int decimals, Unicode::UnicodeChar decimalPoint, int scale) const
{
int length = 0;
if (label < 0 && length < bufferSize - 1)
{
buffer[length++] = '-';
label = -label;
}
if (decimals == 0)
{
Unicode::snprintf(buffer + length, bufferSize - length, "%d", (label + scale / 2) / scale);
}
else if (decimals > 0)
{
Unicode::snprintf(buffer + length, bufferSize - length, "%d", label / scale);
int length = Unicode::strlen(buffer);
if (length < bufferSize - 1)
{
buffer[length++] = decimalPoint;
int32_t remainder = label % scale;
for (int i = 0; i < decimals && length < bufferSize - 1; i++)
{
remainder *= 10;
if (i == decimals - 1 || length == bufferSize - 1)
{
remainder += scale / 2; // Rounding on the last (visible) digit
}
const int digit = (remainder / scale);
buffer[length++] = (Unicode::UnicodeChar)('0' + digit);
remainder %= scale;
}
buffer[length] = (Unicode::UnicodeChar)0;
}
}
}
void GraphLabelsX::draw(const Rect& invalidatedArea) const
{
if (!labelTypedText.hasValidId())
{
return;
}
const Font* fontToDraw = labelTypedText.getFont();
if (!fontToDraw)
{
return;
}
const AbstractDataGraph* graph = getGraph();
const uint8_t alpha = LCD::div255(getAlpha() * graph->getAlpha());
if (alpha == 0)
{
return;
}
const int minorInterval = getIntervalAsInt();
const int majorInterval = (majorLabel == 0) ? 0 : majorLabel->getIntervalAsInt();
if (minorInterval == 0 && majorInterval == 0)
{
drawString(invalidatedArea, fontToDraw, graph, 0, alpha);
}
else if (minorInterval > 0)
{
int rangeMin = graph->getGraphRangeXMin();
int rangeMax = graph->getGraphRangeXMax();
if (rangeMin > rangeMax)
{
const int tmp = rangeMin;
rangeMin = rangeMax;
rangeMax = tmp;
}
const int16_t gapIndex = graph->getGapBeforeIndex();
if (gapIndex <= 0 || gapIndex <= rangeMin || gapIndex > rangeMax)
{
drawIndexRange(invalidatedArea, fontToDraw, graph, rangeMin, rangeMax, alpha);
}
else
{
drawIndexRange(invalidatedArea, fontToDraw, graph, rangeMin, (int)gapIndex - 1, alpha);
drawIndexRange(invalidatedArea, fontToDraw, graph, (int)gapIndex, rangeMax, alpha);
}
}
}
void GraphLabelsX::invalidateGraphPointAt(int16_t index)
{
if (!labelTypedText.hasValidId())
{
return;
}
const Font* fontToDraw = labelTypedText.getFont();
if (!fontToDraw)
{
return;
}
const AbstractDataGraph* graph = getGraph();
const uint8_t alpha = LCD::div255(getAlpha() * graph->getAlpha());
if (alpha == 0)
{
return;
}
const int minorInterval = getIntervalAsInt();
const int majorInterval = (majorLabel == 0) ? 0 : majorLabel->getIntervalAsInt();
bool isOnMinor = (minorInterval > 0 && index == minorInterval * (int)(index / minorInterval));
bool isOnMajor = (majorInterval > 0 && index == majorInterval * (int)(index / majorInterval));
if ((majorInterval == 0 && minorInterval == 0 && index == 0) || (isOnMinor && !isOnMajor))
{
Unicode::UnicodeChar wildcard[20];
formatLabel(wildcard, 20, (graph->indexToGlobalIndex((int)index) * getGraphXAxisScaleScaled(graph)) + getGraphXAxisOffsetScaled(graph), labelDecimals, labelDecimalPoint, graph->getScale());
// Adjust to make label centered
uint16_t labelWidth;
if (labelRotation == TEXT_ROTATE_0 || labelRotation == TEXT_ROTATE_180)
{
labelWidth = fontToDraw->getStringWidth(labelTypedText.getText(), wildcard);
}
else
{
labelWidth = fontToDraw->getMaxTextHeight(labelTypedText.getText(), wildcard) * fontToDraw->getNumberOfLines(labelTypedText.getText(), wildcard) + fontToDraw->getSpacingAbove(labelTypedText.getText(), wildcard);
}
Rect dirty((graph->getGraphAreaMarginLeft() + valueToScreenXQ5(graph, index).round()) - labelWidth / 2, 0, labelWidth, getHeight());
invalidateRect(dirty);
}
}
void GraphLabelsX::drawIndexRange(const Rect& invalidatedArea, const Font* fontToDraw, const AbstractDataGraph* graph, int indexLow, int indexHigh, const uint8_t alpha) const
{
if (indexLow > indexHigh)
{
const int tmp = indexLow;
indexLow = indexHigh;
indexHigh = tmp;
}
// Now indexHigh is higher than indexLow
const int minorInterval = getIntervalAsInt();
if (minorInterval > 0 && abs(indexHigh - indexLow) / minorInterval > 100)
{
return; // Too many labels
}
if (minorInterval == 0)
{
if ((0 >= indexLow && 0 <= indexHigh) || (0 >= indexHigh && 0 <= indexLow))
{
drawString(invalidatedArea, fontToDraw, graph, 0, alpha);
}
return;
}
const int majorInterval = (majorLabel == 0) ? 0 : majorLabel->getIntervalAsInt();
const int minorLo = (int)(indexLow / minorInterval) - 1;
const int minorHi = (int)(indexHigh / minorInterval) + 1;
if (majorInterval == 0)
{
for (int minorIndex = minorLo; minorIndex != minorHi + 1; minorIndex++)
{
const int index = (int)(minorInterval * minorIndex);
if ((index >= indexLow && index <= indexHigh) || (index >= indexHigh && index <= indexLow))
{
drawString(invalidatedArea, fontToDraw, graph, index, alpha);
}
}
}
else
{
const int majorLo = (int)(indexLow / majorInterval) - 1;
const int majorHi = (int)(indexHigh / majorInterval) + 1;
int majorIndex = majorLo;
int32_t majorValue = majorInterval * majorLo;
int32_t minorValue = minorInterval * minorLo;
for (;;)
{
// Draw strings lines up to the major line
while (minorValue < majorValue)
{
if ((minorValue >= indexLow && minorValue <= indexHigh) || (minorValue >= indexHigh && minorValue <= indexLow))
{
drawString(invalidatedArea, fontToDraw, graph, minorValue, alpha);
}
minorValue += minorInterval;
}
// Advance minor past the major line we are about to draw
while (minorValue <= majorValue)
{
minorValue += minorInterval;
}
if (majorValue < minorValue)
{
majorIndex++;
if (majorIndex == majorHi + 1)
{
break;
}
majorValue += majorInterval;
}
}
}
}
void GraphLabelsX::drawString(const Rect& invalidatedArea, const Font* fontToDraw, const AbstractDataGraph* graph, int index, const uint8_t alpha) const
{
const int16_t labelX = valueToScreenXQ5(graph, index).round() - graph->getGraphAreaPaddingLeft();
if (labelX < 0 || labelX >= graph->getGraphAreaWidth())
{
return;
}
Unicode::UnicodeChar wildcard[20];
formatLabel(wildcard, 20, (graph->indexToGlobalIndex(index) * getGraphXAxisScaleScaled(graph)) + getGraphXAxisOffsetScaled(graph), labelDecimals, labelDecimalPoint, graph->getScale());
// Adjust to make label centered
uint16_t labelWidth;
if (labelRotation == TEXT_ROTATE_0 || labelRotation == TEXT_ROTATE_180)
{
labelWidth = fontToDraw->getStringWidth(labelTypedText.getText(), wildcard);
}
else
{
labelWidth = fontToDraw->getMaxTextHeight(labelTypedText.getText(), wildcard) * fontToDraw->getNumberOfLines(labelTypedText.getText(), wildcard) + fontToDraw->getSpacingAbove(labelTypedText.getText(), wildcard);
}
Rect labelRect((graph->getGraphAreaMarginLeft() + valueToScreenXQ5(graph, index).round()) - labelWidth / 2, 0, labelWidth, getHeight());
Rect dirty = labelRect & invalidatedArea;
if (!dirty.isEmpty())
{
dirty.x -= labelRect.x;
dirty.y -= labelRect.y;
translateRectToAbsolute(labelRect);
LCD::StringVisuals visuals(fontToDraw, color, alpha, labelTypedText.getAlignment(), 0, labelRotation, labelTypedText.getTextDirection(), 0, WIDE_TEXT_NONE);
HAL::lcd().drawString(labelRect, dirty, visuals, labelTypedText.getText(), wildcard, 0);
}
}
void GraphLabelsY::draw(const Rect& invalidatedArea) const
{
if (!labelTypedText.hasValidId())
{
return;
}
const Font* fontToDraw = labelTypedText.getFont();
if (!fontToDraw)
{
return;
}
const AbstractDataGraph* graph = getGraph();
const uint8_t alpha = LCD::div255(getAlpha() * graph->getAlpha());
if (alpha == 0)
{
return;
}
const int minorInterval = convertToGraphScale(graph, labelInterval, dataScale);
int majorInterval = getCorrectlyScaledMajorInterval(graph);
if (majorInterval == 0 && minorInterval == 0)
{
drawString(invalidatedArea, fontToDraw, graph, 0, 0, alpha);
}
else if (minorInterval > 0)
{
int rangeMin = getGraphRangeYMinScaled(graph);
int rangeMax = getGraphRangeYMaxScaled(graph);
if (abs(rangeMax - rangeMin) / minorInterval > 100)
{
return; // Too many labels
}
if (rangeMin > rangeMax)
{
const int tmp = rangeMin;
rangeMin = rangeMax;
rangeMax = tmp;
}
// Now rangeMax is higher than rangeMin
const int minorLo = (int)(rangeMin / minorInterval) - 1;
const int minorHi = (int)(rangeMax / minorInterval) + 1;
if (majorInterval == 0)
{
for (int minorIndex = minorLo; minorIndex != minorHi + 1; minorIndex++)
{
drawString(invalidatedArea, fontToDraw, graph, minorInterval * minorIndex, labelInterval * minorIndex, alpha);
}
}
else
{
const int majorLo = (int)(rangeMin / majorInterval) - 1;
const int majorHi = (int)(rangeMax / majorInterval) + 1;
int majorIndex = majorLo;
int32_t majorValue = majorInterval * majorIndex;
int minorIndex = minorLo;
int32_t minorValue = minorInterval * minorIndex;
for (;;)
{
// Draw strings lines up to the major line
while (minorValue < majorValue)
{
drawString(invalidatedArea, fontToDraw, graph, minorValue, labelInterval * minorIndex, alpha);
minorIndex++;
minorValue += minorInterval;
}
// Advance minor past the major line we are about to draw
while (minorValue <= majorValue)
{
minorIndex++;
minorValue += minorInterval;
}
if (majorValue < minorValue)
{
majorIndex++;
if (majorIndex == majorHi + 1)
{
break;
}
majorValue += majorInterval;
}
}
}
}
}
void GraphLabelsY::drawString(const Rect& invalidatedArea, const Font* fontToDraw, const AbstractDataGraph* graph, int valueScaled, int labelScaled, const uint8_t alpha) const
{
const int16_t labelCoord = valueToScreenYQ5(graph, valueScaled).round() - graph->getGraphAreaPaddingTop();
if (labelCoord < 0 || labelCoord >= graph->getGraphAreaHeight())
{
return;
}
Unicode::UnicodeChar wildcard[20];
formatLabel(wildcard, 20, labelScaled, labelDecimals, labelDecimalPoint, dataScale);
// Adjust to make label centered
uint16_t labelHeight;
if (labelRotation == TEXT_ROTATE_0 || labelRotation == TEXT_ROTATE_180)
{
labelHeight = fontToDraw->getMaxTextHeight(labelTypedText.getText(), wildcard) * fontToDraw->getNumberOfLines(labelTypedText.getText(), wildcard) + fontToDraw->getSpacingAbove(labelTypedText.getText(), wildcard);
}
else
{
labelHeight = fontToDraw->getStringWidth(labelTypedText.getText(), wildcard);
}
Rect labelRect(0, (graph->getGraphAreaMarginTop() + valueToScreenYQ5(graph, valueScaled).round()) - labelHeight / 2, getWidth(), labelHeight);
Rect dirty = labelRect & invalidatedArea;
if (!dirty.isEmpty())
{
dirty.x -= labelRect.x;
dirty.y -= labelRect.y;
translateRectToAbsolute(labelRect);
LCD::StringVisuals visuals(fontToDraw, color, alpha, labelTypedText.getAlignment(), 0, labelRotation, labelTypedText.getTextDirection(), 0, WIDE_TEXT_NONE);
HAL::lcd().drawString(labelRect, dirty, visuals, labelTypedText.getText(), wildcard, 0);
}
}
GraphTitle::GraphTitle()
: titleTypedText(TypedText(TYPED_TEXT_INVALID)), titleRotation(TEXT_ROTATE_0)
{
}
void GraphTitle::setTitleTypedText(const TypedText& typedText)
{
titleTypedText = typedText;
}
TypedText GraphTitle::getTitleTypedText() const
{
return titleTypedText;
}
void GraphTitle::setTitleRotation(TextRotation rotation)
{
titleRotation = rotation;
}
TextRotation GraphTitle::getTitleRotation() const
{
return titleRotation;
}
void GraphTitle::draw(const Rect& invalidatedArea) const
{
if (!titleTypedText.hasValidId())
{
return;
}
const Font* fontToDraw = titleTypedText.getFont();
if (!fontToDraw)
{
return;
}
const AbstractDataGraph* graph = getGraph();
const uint8_t alpha = LCD::div255(getAlpha() * graph->getAlpha());
if (alpha == 0)
{
return;
}
const uint16_t lineHeight = fontToDraw->getMaxTextHeight(titleTypedText.getText()) * fontToDraw->getNumberOfLines(titleTypedText.getText()) + fontToDraw->getSpacingAbove(titleTypedText.getText());
Rect labelRect(rect);
// Adjust to make label centered
if (titleRotation == TEXT_ROTATE_0 || titleRotation == TEXT_ROTATE_180)
{
labelRect.y += (labelRect.height - lineHeight) / 2;
labelRect.height = lineHeight;
}
else
{
labelRect.x += (labelRect.width - lineHeight) / 2;
labelRect.width = lineHeight;
}
Rect dirty = labelRect & invalidatedArea;
if (!dirty.isEmpty())
{
dirty.x -= labelRect.x;
dirty.y -= labelRect.y;
translateRectToAbsolute(labelRect);
LCD::StringVisuals visuals(fontToDraw, getColor(), alpha, titleTypedText.getAlignment(), 0, titleRotation, titleTypedText.getTextDirection(), 0, WIDE_TEXT_NONE);
HAL::lcd().drawString(labelRect, dirty, visuals, titleTypedText.getText(), 0, 0);
}
}
bool GraphTitle::drawCanvasWidget(const Rect& /*invalidatedArea*/) const
{
return true;
}
void GraphTitle::invalidateGraphPointAt(int16_t index)
{
}
} // namespace touchgfx

View File

@ -0,0 +1,80 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <touchgfx/widgets/graph/GraphScroll.hpp>
namespace touchgfx
{
DataGraphScroll::DataGraphScroll(int16_t capacity, int* values)
: AbstractDataGraphWithY(capacity, values), current(0)
{
}
void DataGraphScroll::clear()
{
AbstractDataGraphWithY::clear();
current = 0;
}
int32_t DataGraphScroll::indexToGlobalIndex(int16_t index) const
{
if (usedCapacity < maxCapacity)
{
return realIndex(index);
}
return (dataCounter - maxCapacity) + index;
}
void DataGraphScroll::beforeAddValue()
{
if (usedCapacity == maxCapacity)
{
invalidateAllXAxisPoints();
}
}
int16_t DataGraphScroll::addValue(int value)
{
const bool graphFull = usedCapacity == maxCapacity;
const int16_t index = current++;
current %= maxCapacity;
if (index == usedCapacity)
{
usedCapacity++;
}
yValues[index] = value;
if (graphFull)
{
invalidateGraphArea();
invalidateAllXAxisPoints();
}
else
{
invalidateGraphPointAt(index);
}
return index;
}
int16_t DataGraphScroll::realIndex(int16_t index) const
{
return usedCapacity < maxCapacity ? index : (index + current) % maxCapacity;
}
CWRUtil::Q5 DataGraphScroll::indexToXQ5(int16_t index) const
{
return CWRUtil::toQ5(index);
}
} // namespace touchgfx

View File

@ -0,0 +1,55 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <touchgfx/widgets/graph/GraphWrapAndClear.hpp>
namespace touchgfx
{
DataGraphWrapAndClear::DataGraphWrapAndClear(int16_t capacity, int* values)
: AbstractDataGraphWithY(capacity, values)
{
}
int32_t DataGraphWrapAndClear::indexToGlobalIndex(int16_t index) const
{
return (this->dataCounter - this->usedCapacity) + index;
}
void DataGraphWrapAndClear::beforeAddValue()
{
if (usedCapacity >= maxCapacity)
{
invalidateAllXAxisPoints();
clear();
invalidateGraphArea();
}
}
int16_t DataGraphWrapAndClear::addValue(int value)
{
const bool clearGraph = (usedCapacity == 0);
const int16_t index = usedCapacity;
usedCapacity++;
yValues[realIndex(index)] = value;
if (clearGraph)
{
// Label sizes might have grown, also invalidate new sizes
invalidateAllXAxisPoints();
}
invalidateGraphPointAt(index);
return index;
}
} // namespace touchgfx

View File

@ -0,0 +1,98 @@
/**
******************************************************************************
* This file is part of the TouchGFX 4.16.1 distribution.
*
* <h2><center>&copy; Copyright (c) 2021 STMicroelectronics.
* All rights reserved.</center></h2>
*
* This software component is licensed by ST under Ultimate Liberty license
* SLA0044, the "License"; You may not use this file except in compliance with
* the License. You may obtain a copy of the License at:
* www.st.com/SLA0044
*
******************************************************************************
*/
#include <touchgfx/widgets/graph/GraphWrapAndOverwrite.hpp>
namespace touchgfx
{
DataGraphWrapAndOverwrite::DataGraphWrapAndOverwrite(int16_t capacity, int* values)
: AbstractDataGraphWithY(capacity, values), current(0)
{
}
void DataGraphWrapAndOverwrite::clear()
{
AbstractDataGraphWithY::clear();
current = 0;
}
int32_t DataGraphWrapAndOverwrite::indexToGlobalIndex(int16_t index) const
{
if (this->usedCapacity < this->maxCapacity)
{
return index;
}
const int16_t gapIndex = this->getGapBeforeIndex();
if (index < gapIndex)
{
return (this->dataCounter - gapIndex) + index;
}
return ((this->dataCounter - gapIndex) - this->maxCapacity) + index;
}
void DataGraphWrapAndOverwrite::beforeAddValue()
{
if (current == 0 && usedCapacity >= maxCapacity)
{
int xMin = getGraphRangeXMin();
int xMax = getGraphRangeXMax();
for (int i = xMin; i < 0; i++)
{
invalidateXAxisPointAt(i);
}
for (int i = maxCapacity; i <= xMax; i++)
{
invalidateXAxisPointAt(i);
}
}
if (usedCapacity >= maxCapacity)
{
invalidateGraphPointAt(current);
invalidateXAxisPointAt(current);
}
}
int16_t DataGraphWrapAndOverwrite::addValue(int value)
{
const int16_t index = current++;
current %= maxCapacity;
if (index == usedCapacity)
{
usedCapacity++;
}
yValues[realIndex(index)] = value;
setGapBeforeIndex(index + 1);
invalidateGraphPointAt(index);
if (usedCapacity >= maxCapacity)
{
invalidateXAxisPointAt(index);
}
if (index == 0 && usedCapacity >= maxCapacity)
{
int xMin = getGraphRangeXMin();
int xMax = getGraphRangeXMax();
for (int i = xMin; i < 0; i++)
{
invalidateXAxisPointAt(i);
}
for (int i = maxCapacity; i <= xMax; i++)
{
invalidateXAxisPointAt(i);
}
}
return index;
}
} // namespace touchgfx