blob: 213f35a81b888160a383c93d20b46ce9cb2a5426 [file] [log] [blame]
Nader Jawad390d6e82020-09-24 21:35:03 -07001/*
2 * Copyright (C) 2020 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16#include "Bitmap.h"
17#include "GraphicsJNI.h"
18#include "SkImageFilter.h"
19#include "SkImageFilters.h"
20#include "graphics_jni_helpers.h"
21#include "utils/Blur.h"
22#include <utils/Log.h>
23
24using namespace android::uirenderer;
25
26static jlong createOffsetEffect(
27 JNIEnv* env,
28 jobject,
29 jfloat offsetX,
30 jfloat offsetY,
31 jlong inputFilterHandle
32) {
33 auto* inputFilter = reinterpret_cast<const SkImageFilter*>(inputFilterHandle);
34 sk_sp<SkImageFilter> offset = SkImageFilters::Offset(offsetX, offsetY, sk_ref_sp(inputFilter));
35 return reinterpret_cast<jlong>(offset.release());
36}
37
38static jlong createBlurEffect(JNIEnv* env , jobject, jfloat radiusX,
39 jfloat radiusY, jlong inputFilterHandle, jint edgeTreatment) {
40 auto* inputImageFilter = reinterpret_cast<SkImageFilter*>(inputFilterHandle);
41 sk_sp<SkImageFilter> blurFilter =
42 SkImageFilters::Blur(
43 Blur::convertRadiusToSigma(radiusX),
44 Blur::convertRadiusToSigma(radiusY),
45 static_cast<SkTileMode>(edgeTreatment),
46 sk_ref_sp(inputImageFilter),
47 nullptr);
48 return reinterpret_cast<jlong>(blurFilter.release());
49}
50
Nader Jawad0081a9c2020-11-02 19:26:27 -080051static jlong createBitmapEffect(
52 JNIEnv* env,
53 jobject,
54 jlong bitmapHandle,
55 jfloat srcLeft,
56 jfloat srcTop,
57 jfloat srcRight,
58 jfloat srcBottom,
59 jfloat dstLeft,
60 jfloat dstTop,
61 jfloat dstRight,
62 jfloat dstBottom
63) {
64 sk_sp<SkImage> image = android::bitmap::toBitmap(bitmapHandle).makeImage();
65 SkRect srcRect = SkRect::MakeLTRB(srcLeft, srcTop, srcRight, srcBottom);
66 SkRect dstRect = SkRect::MakeLTRB(dstLeft, dstTop, dstRight, dstBottom);
Mike Reed81121542021-02-14 13:20:29 -050067 sk_sp<SkImageFilter> bitmapFilter = SkImageFilters::Image(
68 image, srcRect, dstRect, SkSamplingOptions(SkFilterMode::kLinear));
Nader Jawad0081a9c2020-11-02 19:26:27 -080069 return reinterpret_cast<jlong>(bitmapFilter.release());
70}
71
72static jlong createColorFilterEffect(
73 JNIEnv* env,
74 jobject,
75 jlong colorFilterHandle,
76 jlong inputFilterHandle
77) {
78 auto* colorFilter = reinterpret_cast<const SkColorFilter*>(colorFilterHandle);
79 auto* inputFilter = reinterpret_cast<const SkImageFilter*>(inputFilterHandle);
80 sk_sp<SkImageFilter> colorFilterImageFilter = SkImageFilters::ColorFilter(
81 sk_ref_sp(colorFilter), sk_ref_sp(inputFilter), nullptr);
82 return reinterpret_cast<jlong>(colorFilterImageFilter.release());
83}
84
85static jlong createBlendModeEffect(
86 JNIEnv* env,
87 jobject,
88 jlong backgroundImageFilterHandle,
89 jlong foregroundImageFilterHandle,
90 jint blendmodeHandle
91) {
92 auto* backgroundFilter = reinterpret_cast<const SkImageFilter*>(backgroundImageFilterHandle);
93 auto* foregroundFilter = reinterpret_cast<const SkImageFilter*>(foregroundImageFilterHandle);
94 SkBlendMode blendMode = static_cast<SkBlendMode>(blendmodeHandle);
95 sk_sp<SkImageFilter> xfermodeFilter = SkImageFilters::Blend(
96 blendMode,
97 sk_ref_sp(backgroundFilter),
98 sk_ref_sp(foregroundFilter)
99 );
100 return reinterpret_cast<jlong>(xfermodeFilter.release());
101}
102
103static jlong createChainEffect(
104 JNIEnv* env,
105 jobject,
106 jlong outerFilterHandle,
107 jlong innerFilterHandle
108) {
109 auto* outerImageFilter = reinterpret_cast<const SkImageFilter*>(outerFilterHandle);
110 auto* innerImageFilter = reinterpret_cast<const SkImageFilter*>(innerFilterHandle);
111 sk_sp<SkImageFilter> composeFilter = SkImageFilters::Compose(
112 sk_ref_sp(outerImageFilter),
113 sk_ref_sp(innerImageFilter)
114 );
115 return reinterpret_cast<jlong>(composeFilter.release());
116}
117
Nader Jawad011aac82021-01-25 18:02:07 -0800118static jlong createShaderEffect(
119 JNIEnv* env,
120 jobject,
121 jlong shaderHandle
122) {
123 auto* shader = reinterpret_cast<const SkShader*>(shaderHandle);
124 sk_sp<SkImageFilter> shaderFilter = SkImageFilters::Shader(
125 sk_ref_sp(shader), nullptr
126 );
127 return reinterpret_cast<jlong>(shaderFilter.release());
128}
129
Derek Sollenberger56c07982021-11-05 19:25:05 +0000130static inline int ThrowIAEFmt(JNIEnv* env, const char* fmt, ...) {
131 va_list args;
132 va_start(args, fmt);
133 int ret = jniThrowExceptionFmt(env, "java/lang/IllegalArgumentException", fmt, args);
134 va_end(args);
135 return ret;
136}
137
138static jlong createRuntimeShaderEffect(JNIEnv* env, jobject, jlong shaderBuilderHandle,
139 jstring inputShaderName) {
140 SkRuntimeShaderBuilder* builder =
141 reinterpret_cast<SkRuntimeShaderBuilder*>(shaderBuilderHandle);
142 ScopedUtfChars name(env, inputShaderName);
143
144 if (builder->child(name.c_str()).fChild == nullptr) {
145 ThrowIAEFmt(env,
146 "unable to find a uniform with the name '%s' of the correct "
147 "type defined by the provided RuntimeShader",
148 name.c_str());
149 return 0;
150 }
151
152 sk_sp<SkImageFilter> filter = SkImageFilters::RuntimeShader(*builder, name.c_str(), nullptr);
153 return reinterpret_cast<jlong>(filter.release());
154}
155
Nader Jawad390d6e82020-09-24 21:35:03 -0700156static void RenderEffect_safeUnref(SkImageFilter* filter) {
157 SkSafeUnref(filter);
158}
159
160static jlong getRenderEffectFinalizer(JNIEnv*, jobject) {
161 return static_cast<jlong>(reinterpret_cast<uintptr_t>(&RenderEffect_safeUnref));
162}
163
164static const JNINativeMethod gRenderEffectMethods[] = {
Derek Sollenberger56c07982021-11-05 19:25:05 +0000165 {"nativeGetFinalizer", "()J", (void*)getRenderEffectFinalizer},
166 {"nativeCreateOffsetEffect", "(FFJ)J", (void*)createOffsetEffect},
167 {"nativeCreateBlurEffect", "(FFJI)J", (void*)createBlurEffect},
168 {"nativeCreateBitmapEffect", "(JFFFFFFFF)J", (void*)createBitmapEffect},
169 {"nativeCreateColorFilterEffect", "(JJ)J", (void*)createColorFilterEffect},
170 {"nativeCreateBlendModeEffect", "(JJI)J", (void*)createBlendModeEffect},
171 {"nativeCreateChainEffect", "(JJ)J", (void*)createChainEffect},
172 {"nativeCreateShaderEffect", "(J)J", (void*)createShaderEffect},
173 {"nativeCreateRuntimeShaderEffect", "(JLjava/lang/String;)J",
174 (void*)createRuntimeShaderEffect}};
Nader Jawad390d6e82020-09-24 21:35:03 -0700175
176int register_android_graphics_RenderEffect(JNIEnv* env) {
177 android::RegisterMethodsOrDie(env, "android/graphics/RenderEffect",
178 gRenderEffectMethods, NELEM(gRenderEffectMethods));
179 return 0;
Mike Reed81121542021-02-14 13:20:29 -0500180}