blob: c06e36681bf6af6430318486c799373c47974988 [file] [log] [blame]
epoger@google.comec3ed6a2011-07-28 14:26:00 +00001/*
2 * Copyright 2011 Google Inc.
3 *
4 * Use of this source code is governed by a BSD-style license that can be
5 * found in the LICENSE file.
6 */
tfarina@chromium.orge4fafb12013-12-12 21:11:12 +00007
Mike Kleinc0bd9f92019-04-23 12:05:21 -05008#include "include/core/SkBlendMode.h"
9#include "include/core/SkColor.h"
10#include "include/core/SkColorFilter.h"
Mike Klein6bba1bc2021-02-16 11:59:00 -060011#include "include/core/SkColorSpace.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050012#include "include/core/SkRefCnt.h"
13#include "include/core/SkTypes.h"
Mike Klein6bba1bc2021-02-16 11:59:00 -060014#include "include/effects/SkColorMatrix.h"
Mike Kleinc0bd9f92019-04-23 12:05:21 -050015#include "include/utils/SkRandom.h"
16#include "src/core/SkAutoMalloc.h"
17#include "src/core/SkReadBuffer.h"
18#include "src/core/SkWriteBuffer.h"
19#include "tests/Test.h"
reed@google.com43c50c82011-04-14 15:50:52 +000020
Ben Wagnereed61282018-04-17 14:14:51 -040021class SkFlattenable;
22
reedd053ce92016-03-22 10:17:23 -070023static sk_sp<SkColorFilter> reincarnate_colorfilter(SkFlattenable* obj) {
brianosmanfad98562016-05-04 11:06:28 -070024 SkBinaryWriteBuffer wb;
djsollen@google.comcefc8652012-03-26 15:52:10 +000025 wb.writeFlattenable(obj);
reed@google.com43c50c82011-04-14 15:50:52 +000026
commit-bot@chromium.org8b0e8ac2014-01-30 18:58:24 +000027 size_t size = wb.bytesWritten();
reed@google.com43c50c82011-04-14 15:50:52 +000028 SkAutoSMalloc<1024> storage(size);
29 // make a copy into storage
djsollen@google.comc73dd5c2012-08-07 15:54:32 +000030 wb.writeToMemory(storage.get());
reed@google.com43c50c82011-04-14 15:50:52 +000031
commit-bot@chromium.org8b0e8ac2014-01-30 18:58:24 +000032 SkReadBuffer rb(storage.get(), size);
reed@google.com35348222013-10-16 13:05:06 +000033 return rb.readColorFilter();
reed@google.com43c50c82011-04-14 15:50:52 +000034}
35
36///////////////////////////////////////////////////////////////////////////////
37
Mike Reed7d954ad2016-10-28 15:42:34 -040038#define ILLEGAL_MODE ((SkBlendMode)-1)
reed@google.com43c50c82011-04-14 15:50:52 +000039
tfarina@chromium.orge4fafb12013-12-12 21:11:12 +000040DEF_TEST(ColorFilter, reporter) {
commit-bot@chromium.orge0e7cfe2013-09-09 20:09:12 +000041 SkRandom rand;
reed@google.com43c50c82011-04-14 15:50:52 +000042
Mike Reed7d954ad2016-10-28 15:42:34 -040043 for (int mode = 0; mode <= (int)SkBlendMode::kLastMode; mode++) {
reed@google.com43c50c82011-04-14 15:50:52 +000044 SkColor color = rand.nextU();
45
46 // ensure we always get a filter, by avoiding the possibility of a
halcanary96fcdcc2015-08-27 07:41:13 -070047 // special case that would return nullptr (if color's alpha is 0 or 0xFF)
reed@google.com43c50c82011-04-14 15:50:52 +000048 color = SkColorSetA(color, 0x7F);
49
Mike Reedb286bc22019-04-08 16:23:20 -040050 auto cf = SkColorFilters::Blend(color, (SkBlendMode)mode);
reed@google.com43c50c82011-04-14 15:50:52 +000051
52 // allow for no filter if we're in Dst mode (its a no op)
Mike Reed7d954ad2016-10-28 15:42:34 -040053 if (SkBlendMode::kDst == (SkBlendMode)mode && nullptr == cf) {
reed@google.com43c50c82011-04-14 15:50:52 +000054 continue;
55 }
56
reed@google.com43c50c82011-04-14 15:50:52 +000057 REPORTER_ASSERT(reporter, cf);
58
59 SkColor c = ~color;
Mike Reed7d954ad2016-10-28 15:42:34 -040060 SkBlendMode m = ILLEGAL_MODE;
reed@google.com43c50c82011-04-14 15:50:52 +000061
62 SkColor expectedColor = color;
Mike Reed7d954ad2016-10-28 15:42:34 -040063 SkBlendMode expectedMode = (SkBlendMode)mode;
reed@google.com43c50c82011-04-14 15:50:52 +000064
65// SkDebugf("--- mc [%d %x] ", mode, color);
66
Mike Reed4e1b07e2019-04-25 21:04:49 -040067 REPORTER_ASSERT(reporter, cf->asAColorMode(&c, (SkBlendMode*)&m));
reed@google.com43c50c82011-04-14 15:50:52 +000068 // handle special-case folding by the factory
Mike Reed7d954ad2016-10-28 15:42:34 -040069 if (SkBlendMode::kClear == (SkBlendMode)mode) {
reed@google.com43c50c82011-04-14 15:50:52 +000070 if (c != expectedColor) {
71 expectedColor = 0;
72 }
73 if (m != expectedMode) {
Mike Reed7d954ad2016-10-28 15:42:34 -040074 expectedMode = SkBlendMode::kSrc;
reed@google.com43c50c82011-04-14 15:50:52 +000075 }
rmistry@google.comd6176b02012-08-23 18:14:13 +000076 }
reed@google.com43c50c82011-04-14 15:50:52 +000077
78// SkDebugf("--- got [%d %x] expected [%d %x]\n", m, c, expectedMode, expectedColor);
79
80 REPORTER_ASSERT(reporter, c == expectedColor);
81 REPORTER_ASSERT(reporter, m == expectedMode);
rmistry@google.comd6176b02012-08-23 18:14:13 +000082
reed@google.com43c50c82011-04-14 15:50:52 +000083 {
reedd053ce92016-03-22 10:17:23 -070084 auto cf2 = reincarnate_colorfilter(cf.get());
reed@google.com43c50c82011-04-14 15:50:52 +000085 REPORTER_ASSERT(reporter, cf2);
86
87 SkColor c2 = ~color;
Mike Reed7d954ad2016-10-28 15:42:34 -040088 SkBlendMode m2 = ILLEGAL_MODE;
Mike Reed4e1b07e2019-04-25 21:04:49 -040089 REPORTER_ASSERT(reporter, cf2->asAColorMode(&c2, (SkBlendMode*)&m2));
reed@google.com43c50c82011-04-14 15:50:52 +000090 REPORTER_ASSERT(reporter, c2 == expectedColor);
91 REPORTER_ASSERT(reporter, m2 == expectedMode);
92 }
93 }
94}
Mike Klein6bba1bc2021-02-16 11:59:00 -060095
96DEF_TEST(WorkingFormatFilterFlags, r) {
97 {
98 // A matrix with final row 0,0,0,1,0 shouldn't change alpha.
99 sk_sp<SkColorFilter> cf = SkColorFilters::Matrix({1,0,0,0,0,
100 0,1,0,0,0,
101 0,0,1,0,0,
102 0,0,0,1,0});
103 REPORTER_ASSERT(r, cf->isAlphaUnchanged());
104
105 // No working format change will itself change alpha.
106 SkAlphaType unpremul = kUnpremul_SkAlphaType;
107 cf = SkColorFilters::WithWorkingFormat(std::move(cf),
108 &SkNamedTransferFn::kLinear,
109 &SkNamedGamut::kDisplayP3,
110 &unpremul);
111 REPORTER_ASSERT(r, cf->isAlphaUnchanged());
112 }
113
114 {
115 // Here's a matrix that definitely does change alpha.
116 sk_sp<SkColorFilter> cf = SkColorFilters::Matrix({1,0,0,0,0,
117 0,1,0,0,0,
118 0,0,1,0,0,
119 0,0,0,0,1});
120 REPORTER_ASSERT(r, !cf->isAlphaUnchanged());
121
122 SkAlphaType unpremul = kUnpremul_SkAlphaType;
123 cf = SkColorFilters::WithWorkingFormat(std::move(cf),
124 &SkNamedTransferFn::kLinear,
125 &SkNamedGamut::kDisplayP3,
126 &unpremul);
127 REPORTER_ASSERT(r, !cf->isAlphaUnchanged());
128 }
129}