1 /*
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 "SkBlendMode.h"
19 #include "SkImageFilter.h"
20 #include "SkImageFilters.h"
21 #include "graphics_jni_helpers.h"
22 #include "utils/Blur.h"
23 #include <utils/Log.h>
24
25 using namespace android::uirenderer;
26
createOffsetEffect(JNIEnv * env,jobject,jfloat offsetX,jfloat offsetY,jlong inputFilterHandle)27 static jlong createOffsetEffect(
28 JNIEnv* env,
29 jobject,
30 jfloat offsetX,
31 jfloat offsetY,
32 jlong inputFilterHandle
33 ) {
34 auto* inputFilter = reinterpret_cast<const SkImageFilter*>(inputFilterHandle);
35 sk_sp<SkImageFilter> offset = SkImageFilters::Offset(offsetX, offsetY, sk_ref_sp(inputFilter));
36 return reinterpret_cast<jlong>(offset.release());
37 }
38
createBlurEffect(JNIEnv * env,jobject,jfloat radiusX,jfloat radiusY,jlong inputFilterHandle,jint edgeTreatment)39 static jlong createBlurEffect(JNIEnv* env , jobject, jfloat radiusX,
40 jfloat radiusY, jlong inputFilterHandle, jint edgeTreatment) {
41 auto* inputImageFilter = reinterpret_cast<SkImageFilter*>(inputFilterHandle);
42 sk_sp<SkImageFilter> blurFilter =
43 SkImageFilters::Blur(
44 Blur::convertRadiusToSigma(radiusX),
45 Blur::convertRadiusToSigma(radiusY),
46 static_cast<SkTileMode>(edgeTreatment),
47 sk_ref_sp(inputImageFilter),
48 nullptr);
49 return reinterpret_cast<jlong>(blurFilter.release());
50 }
51
createBitmapEffect(JNIEnv * env,jobject,jlong bitmapHandle,jfloat srcLeft,jfloat srcTop,jfloat srcRight,jfloat srcBottom,jfloat dstLeft,jfloat dstTop,jfloat dstRight,jfloat dstBottom)52 static jlong createBitmapEffect(
53 JNIEnv* env,
54 jobject,
55 jlong bitmapHandle,
56 jfloat srcLeft,
57 jfloat srcTop,
58 jfloat srcRight,
59 jfloat srcBottom,
60 jfloat dstLeft,
61 jfloat dstTop,
62 jfloat dstRight,
63 jfloat dstBottom
64 ) {
65 sk_sp<SkImage> image = android::bitmap::toBitmap(bitmapHandle).makeImage();
66 SkRect srcRect = SkRect::MakeLTRB(srcLeft, srcTop, srcRight, srcBottom);
67 SkRect dstRect = SkRect::MakeLTRB(dstLeft, dstTop, dstRight, dstBottom);
68 sk_sp<SkImageFilter> bitmapFilter = SkImageFilters::Image(
69 image, srcRect, dstRect, SkSamplingOptions(SkFilterMode::kLinear));
70 return reinterpret_cast<jlong>(bitmapFilter.release());
71 }
72
createColorFilterEffect(JNIEnv * env,jobject,jlong colorFilterHandle,jlong inputFilterHandle)73 static jlong createColorFilterEffect(
74 JNIEnv* env,
75 jobject,
76 jlong colorFilterHandle,
77 jlong inputFilterHandle
78 ) {
79 auto* colorFilter = reinterpret_cast<const SkColorFilter*>(colorFilterHandle);
80 auto* inputFilter = reinterpret_cast<const SkImageFilter*>(inputFilterHandle);
81 sk_sp<SkImageFilter> colorFilterImageFilter = SkImageFilters::ColorFilter(
82 sk_ref_sp(colorFilter), sk_ref_sp(inputFilter), nullptr);
83 return reinterpret_cast<jlong>(colorFilterImageFilter.release());
84 }
85
createBlendModeEffect(JNIEnv * env,jobject,jlong backgroundImageFilterHandle,jlong foregroundImageFilterHandle,jint blendmodeHandle)86 static jlong createBlendModeEffect(
87 JNIEnv* env,
88 jobject,
89 jlong backgroundImageFilterHandle,
90 jlong foregroundImageFilterHandle,
91 jint blendmodeHandle
92 ) {
93 auto* backgroundFilter = reinterpret_cast<const SkImageFilter*>(backgroundImageFilterHandle);
94 auto* foregroundFilter = reinterpret_cast<const SkImageFilter*>(foregroundImageFilterHandle);
95 SkBlendMode blendMode = static_cast<SkBlendMode>(blendmodeHandle);
96 sk_sp<SkImageFilter> xfermodeFilter = SkImageFilters::Blend(
97 blendMode,
98 sk_ref_sp(backgroundFilter),
99 sk_ref_sp(foregroundFilter)
100 );
101 return reinterpret_cast<jlong>(xfermodeFilter.release());
102 }
103
createChainEffect(JNIEnv * env,jobject,jlong outerFilterHandle,jlong innerFilterHandle)104 static jlong createChainEffect(
105 JNIEnv* env,
106 jobject,
107 jlong outerFilterHandle,
108 jlong innerFilterHandle
109 ) {
110 auto* outerImageFilter = reinterpret_cast<const SkImageFilter*>(outerFilterHandle);
111 auto* innerImageFilter = reinterpret_cast<const SkImageFilter*>(innerFilterHandle);
112 sk_sp<SkImageFilter> composeFilter = SkImageFilters::Compose(
113 sk_ref_sp(outerImageFilter),
114 sk_ref_sp(innerImageFilter)
115 );
116 return reinterpret_cast<jlong>(composeFilter.release());
117 }
118
createShaderEffect(JNIEnv * env,jobject,jlong shaderHandle)119 static jlong createShaderEffect(
120 JNIEnv* env,
121 jobject,
122 jlong shaderHandle
123 ) {
124 auto* shader = reinterpret_cast<const SkShader*>(shaderHandle);
125 sk_sp<SkImageFilter> shaderFilter = SkImageFilters::Shader(
126 sk_ref_sp(shader), nullptr
127 );
128 return reinterpret_cast<jlong>(shaderFilter.release());
129 }
130
ThrowIAEFmt(JNIEnv * env,const char * fmt,...)131 static inline int ThrowIAEFmt(JNIEnv* env, const char* fmt, ...) {
132 va_list args;
133 va_start(args, fmt);
134 int ret = jniThrowExceptionFmt(env, "java/lang/IllegalArgumentException", fmt, args);
135 va_end(args);
136 return ret;
137 }
138
createRuntimeShaderEffect(JNIEnv * env,jobject,jlong shaderBuilderHandle,jstring inputShaderName)139 static jlong createRuntimeShaderEffect(JNIEnv* env, jobject, jlong shaderBuilderHandle,
140 jstring inputShaderName) {
141 SkRuntimeShaderBuilder* builder =
142 reinterpret_cast<SkRuntimeShaderBuilder*>(shaderBuilderHandle);
143 ScopedUtfChars name(env, inputShaderName);
144
145 if (builder->child(name.c_str()).fChild == nullptr) {
146 ThrowIAEFmt(env,
147 "unable to find a uniform with the name '%s' of the correct "
148 "type defined by the provided RuntimeShader",
149 name.c_str());
150 return 0;
151 }
152
153 sk_sp<SkImageFilter> filter = SkImageFilters::RuntimeShader(*builder, name.c_str(), nullptr);
154 return reinterpret_cast<jlong>(filter.release());
155 }
156
RenderEffect_safeUnref(SkImageFilter * filter)157 static void RenderEffect_safeUnref(SkImageFilter* filter) {
158 SkSafeUnref(filter);
159 }
160
getRenderEffectFinalizer(JNIEnv *,jobject)161 static jlong getRenderEffectFinalizer(JNIEnv*, jobject) {
162 return static_cast<jlong>(reinterpret_cast<uintptr_t>(&RenderEffect_safeUnref));
163 }
164
165 static const JNINativeMethod gRenderEffectMethods[] = {
166 {"nativeGetFinalizer", "()J", (void*)getRenderEffectFinalizer},
167 {"nativeCreateOffsetEffect", "(FFJ)J", (void*)createOffsetEffect},
168 {"nativeCreateBlurEffect", "(FFJI)J", (void*)createBlurEffect},
169 {"nativeCreateBitmapEffect", "(JFFFFFFFF)J", (void*)createBitmapEffect},
170 {"nativeCreateColorFilterEffect", "(JJ)J", (void*)createColorFilterEffect},
171 {"nativeCreateBlendModeEffect", "(JJI)J", (void*)createBlendModeEffect},
172 {"nativeCreateChainEffect", "(JJ)J", (void*)createChainEffect},
173 {"nativeCreateShaderEffect", "(J)J", (void*)createShaderEffect},
174 {"nativeCreateRuntimeShaderEffect", "(JLjava/lang/String;)J",
175 (void*)createRuntimeShaderEffect}};
176
register_android_graphics_RenderEffect(JNIEnv * env)177 int register_android_graphics_RenderEffect(JNIEnv* env) {
178 android::RegisterMethodsOrDie(env, "android/graphics/RenderEffect",
179 gRenderEffectMethods, NELEM(gRenderEffectMethods));
180 return 0;
181 }
182