mirror of
https://github.com/FFmpeg/FFmpeg.git
synced 2025-01-13 21:28:01 +02:00
lavc/dvdsubenc: improve color distance function.
Consider the color space as an hypercone with apex alpha=0 and base alpha=1 instead of an hypercube. Make the encoder consider very transparent colors more similar even if the hue is very different. This corresponds roughly to using the alpha as a weight for the color difference. Only 4 bits of alpha are used, because this is what dvdsub uses, and it avoids overflows. Fix trac ticket #2005.
This commit is contained in:
parent
8dbbaf568e
commit
5ed5e90f2a
@ -94,10 +94,14 @@ static void dvd_encode_rle(uint8_t **pq,
|
|||||||
static int color_distance(uint32_t a, uint32_t b)
|
static int color_distance(uint32_t a, uint32_t b)
|
||||||
{
|
{
|
||||||
int r = 0, d, i;
|
int r = 0, d, i;
|
||||||
|
int alpha_a = 8, alpha_b = 8;
|
||||||
|
|
||||||
for (i = 0; i < 32; i += 8) {
|
for (i = 24; i >= 0; i -= 8) {
|
||||||
d = ((a >> i) & 0xFF) - ((b >> i) & 0xFF);
|
d = alpha_a * (int)((a >> i) & 0xFF) -
|
||||||
|
alpha_b * (int)((b >> i) & 0xFF);
|
||||||
r += d * d;
|
r += d * d;
|
||||||
|
alpha_a = a >> 28;
|
||||||
|
alpha_b = b >> 28;
|
||||||
}
|
}
|
||||||
return r;
|
return r;
|
||||||
}
|
}
|
||||||
@ -130,7 +134,8 @@ static void count_colors(AVCodecContext *avctx, unsigned hits[33],
|
|||||||
if (match) {
|
if (match) {
|
||||||
best_d = INT_MAX;
|
best_d = INT_MAX;
|
||||||
for (j = 0; j < 16; j++) {
|
for (j = 0; j < 16; j++) {
|
||||||
d = color_distance(color & 0xFFFFFF, dvdc->global_palette[j]);
|
d = color_distance(0xFF000000 | color,
|
||||||
|
0xFF000000 | dvdc->global_palette[j]);
|
||||||
if (d < best_d) {
|
if (d < best_d) {
|
||||||
best_d = d;
|
best_d = d;
|
||||||
best_j = j;
|
best_j = j;
|
||||||
|
Loading…
Reference in New Issue
Block a user