QSkinny 0.8.0
C++/Qt UI toolkit
Loading...
Searching...
No Matches
QskColorRamp.cpp
1/******************************************************************************
2 * QSkinny - Copyright (C) The authors
3 * SPDX-License-Identifier: BSD-3-Clause
4 *****************************************************************************/
5
6#include "QskColorRamp.h"
7#include "QskRgbValue.h"
8#include "QskInternalMacros.h"
9
10QSK_QT_PRIVATE_BEGIN
11#include <private/qrhi_p.h>
12#include <private/qsgplaintexture_p.h>
13QSK_QT_PRIVATE_END
14
15#include <qcoreapplication.h>
16
17namespace
18{
19 class Texture : public QSGPlainTexture
20 {
21 public:
22 Texture( const QskGradientStops& stops, QskGradient::SpreadMode spreadMode )
23 {
24 /*
25 Qt creates tables of 1024 colors, while Chrome, Firefox, and Android
26 seem to use 256 colors only ( according to maybe outdated sources
27 from the internet ),
28 */
29
30 const int size = qBound( 256, 2 * stops.count(), 1024 );
31 setImage( QskRgb::colorTable( size, stops ) );
32
33 const auto wrapMode = this->wrapMode( spreadMode );
34
35 setHorizontalWrapMode( wrapMode );
36 setVerticalWrapMode( wrapMode );
37
38 setFiltering( QSGTexture::Linear );
39 }
40
41 private:
42 static inline QSGTexture::WrapMode wrapMode( QskGradient::SpreadMode spreadMode )
43 {
44 switch ( spreadMode )
45 {
46 case QskGradient::RepeatSpread:
47 return QSGTexture::Repeat;
48
49 case QskGradient::ReflectSpread:
50 return QSGTexture::MirroredRepeat;
51
52 default:
53 return QSGTexture::ClampToEdge;
54 }
55 }
56 };
57
58 class HashKey
59 {
60 public:
61 inline bool operator==( const HashKey& other ) const
62 {
63 return rhi == other.rhi && spreadMode == other.spreadMode && stops == other.stops;
64 }
65
66 const void* rhi;
67 const QskGradientStops stops;
68 const QskGradient::SpreadMode spreadMode;
69 };
70
71 inline size_t qHash( const HashKey& key, size_t seed = 0 )
72 {
73 size_t values = seed + key.spreadMode;
74
75 for ( const auto& stop : key.stops )
76 values += stop.rgb();
77
78 return values;
79 }
80
81 class Cache
82 {
83 public:
84 ~Cache() { qDeleteAll( m_hashTable ); }
85
86 void cleanupRhi( const QRhi* );
87
88 Texture* texture( const void* rhi,
89 const QskGradientStops&, QskGradient::SpreadMode );
90
91 private:
92 QHash< HashKey, Texture* > m_hashTable;
93 QVector< const QRhi* > m_rhiTable; // no QSet: we usually have only one entry
94 };
95
96 static Cache* s_cache;
97}
98
99static void qskCleanupCache()
100{
101 delete s_cache;
102 s_cache = nullptr;
103}
104
105static void qskCleanupRhi( const QRhi* rhi )
106{
107 if ( s_cache )
108 s_cache->cleanupRhi( rhi );
109}
110
111Texture* Cache::texture( const void* rhi,
112 const QskGradientStops& stops, QskGradient::SpreadMode spreadMode )
113{
114 const HashKey key { rhi, stops, spreadMode };
115
116 auto texture = m_hashTable[key];
117 if ( texture == nullptr )
118 {
119 texture = new Texture( stops, spreadMode );
120 m_hashTable[ key ] = texture;
121
122 if ( rhi != nullptr )
123 {
124 auto myrhi = ( QRhi* )rhi;
125
126 if ( !m_rhiTable.contains( myrhi ) )
127 {
128 myrhi->addCleanupCallback( qskCleanupRhi );
129 m_rhiTable += myrhi;
130 }
131 }
132 }
133
134 return texture;
135}
136
137void Cache::cleanupRhi( const QRhi* rhi )
138{
139 for ( auto it = m_hashTable.begin(); it != m_hashTable.end(); )
140 {
141 if ( it.key().rhi == rhi )
142 {
143 delete it.value();
144 it = m_hashTable.erase( it );
145 }
146 else
147 {
148 ++it;
149 }
150 }
151
152 m_rhiTable.removeAll( rhi );
153}
154
155QSGTexture* QskColorRamp::texture( const void* rhi,
156 const QskGradientStops& stops, QskGradient::SpreadMode spreadMode )
157{
158 if ( s_cache == nullptr )
159 {
160 s_cache = new Cache();
161
162 /*
163 For RHI we have QRhi::addCleanupCallback, but with
164 OpenGL we would have to fiddle around with QOpenGLSharedResource
165 But as the OpenGL path is only for Qt5 we do not want to spend
166 much energy on finetuning the resource management.
167 */
168 qAddPostRoutine( qskCleanupCache );
169 }
170
171 return s_cache->texture( rhi, stops, spreadMode );
172}