epoger@google.com | ec3ed6a | 2011-07-28 14:26:00 +0000 | [diff] [blame] | 1 | /* |
| 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 | */ |
commit-bot@chromium.org | c3bd8af | 2014-02-13 17:14:46 +0000 | [diff] [blame] | 7 | |
reed@android.com | 00dae86 | 2009-06-10 15:38:48 +0000 | [diff] [blame] | 8 | #ifndef skiagm_DEFINED |
| 9 | #define skiagm_DEFINED |
| 10 | |
Ben Wagner | 7fde8e1 | 2019-05-01 17:28:53 -0400 | [diff] [blame] | 11 | #include "include/core/SkColor.h" |
| 12 | #include "include/core/SkScalar.h" |
Mike Klein | c0bd9f9 | 2019-04-23 12:05:21 -0500 | [diff] [blame] | 13 | #include "include/core/SkSize.h" |
| 14 | #include "include/core/SkString.h" |
Ben Wagner | 7fde8e1 | 2019-05-01 17:28:53 -0400 | [diff] [blame] | 15 | #include "include/core/SkTypes.h" |
Mike Klein | c0bd9f9 | 2019-04-23 12:05:21 -0500 | [diff] [blame] | 16 | #include "include/private/SkMacros.h" |
Mike Klein | c0bd9f9 | 2019-04-23 12:05:21 -0500 | [diff] [blame] | 17 | #include "tools/Registry.h" |
reed@android.com | 00dae86 | 2009-06-10 15:38:48 +0000 | [diff] [blame] | 18 | |
Hal Canary | edda565 | 2019-08-05 10:28:09 -0400 | [diff] [blame] | 19 | #include <memory> |
| 20 | |
Ben Wagner | 7fde8e1 | 2019-05-01 17:28:53 -0400 | [diff] [blame] | 21 | class GrContext; |
| 22 | class GrRenderTargetContext; |
| 23 | class SkCanvas; |
Mike Reed | 88f5671 | 2019-02-04 16:50:10 -0500 | [diff] [blame] | 24 | class SkMetaData; |
bsalomon | 4ee6bd8 | 2015-05-27 13:23:23 -0700 | [diff] [blame] | 25 | struct GrContextOptions; |
mtklein@google.com | 62b50b7 | 2013-09-16 20:42:15 +0000 | [diff] [blame] | 26 | |
Hal Canary | edda565 | 2019-08-05 10:28:09 -0400 | [diff] [blame] | 27 | #define DEF_GM(CODE) \ |
| 28 | static skiagm::GMRegistry SK_MACRO_APPEND_LINE(REG_)(\ |
| 29 | [](){return std::unique_ptr<skiagm::GM>([](){ CODE ; }());}); |
reed@google.com | 4117a24 | 2012-10-30 20:26:58 +0000 | [diff] [blame] | 30 | |
Chris Dalton | 3a77837 | 2019-02-07 15:23:36 -0700 | [diff] [blame] | 31 | // A Simple GM is a rendering test that does not store state between rendering calls or make use of |
| 32 | // the onOnceBeforeDraw() virtual; it consists of: |
halcanary | 2a24338 | 2015-09-09 08:16:41 -0700 | [diff] [blame] | 33 | // * A name. |
| 34 | // * Prefered width and height. |
| 35 | // * Optionally, a background color (default is white). |
Chris Dalton | 3a77837 | 2019-02-07 15:23:36 -0700 | [diff] [blame] | 36 | // * A standalone function pointer that implements its onDraw method. |
halcanary | 2a24338 | 2015-09-09 08:16:41 -0700 | [diff] [blame] | 37 | #define DEF_SIMPLE_GM(NAME, CANVAS, W, H) \ |
| 38 | DEF_SIMPLE_GM_BG_NAME(NAME, CANVAS, W, H, SK_ColorWHITE, SkString(#NAME)) |
Chris Dalton | 3a77837 | 2019-02-07 15:23:36 -0700 | [diff] [blame] | 39 | #define DEF_SIMPLE_GM_BG(NAME, CANVAS, W, H, BGCOLOR) \ |
halcanary | 2a24338 | 2015-09-09 08:16:41 -0700 | [diff] [blame] | 40 | DEF_SIMPLE_GM_BG_NAME(NAME, CANVAS, W, H, BGCOLOR, SkString(#NAME)) |
Chris Dalton | 3a77837 | 2019-02-07 15:23:36 -0700 | [diff] [blame] | 41 | #define DEF_SIMPLE_GM_BG_NAME(NAME, CANVAS, W, H, BGCOLOR, NAME_STR) \ |
Chris Dalton | 50e24d7 | 2019-02-07 16:20:09 -0700 | [diff] [blame] | 42 | static void SK_MACRO_CONCAT(NAME,_GM_inner)(SkCanvas*); \ |
| 43 | DEF_SIMPLE_GM_BG_NAME_CAN_FAIL(NAME, CANVAS,, W, H, BGCOLOR, NAME_STR) { \ |
| 44 | SK_MACRO_CONCAT(NAME,_GM_inner)(CANVAS); \ |
| 45 | return skiagm::DrawResult::kOk; \ |
| 46 | } \ |
| 47 | void SK_MACRO_CONCAT(NAME,_GM_inner)(SkCanvas* CANVAS) |
| 48 | |
| 49 | #define DEF_SIMPLE_GM_CAN_FAIL(NAME, CANVAS, ERR_MSG, W, H) \ |
| 50 | DEF_SIMPLE_GM_BG_NAME_CAN_FAIL(NAME, CANVAS, ERR_MSG, W, H, SK_ColorWHITE, SkString(#NAME)) |
| 51 | #define DEF_SIMPLE_GM_BG_CAN_FAIL(NAME, CANVAS, ERR_MSG, W, H, BGCOLOR) \ |
| 52 | DEF_SIMPLE_GM_BG_NAME_CAN_FAIL(NAME, CANVAS, ERR_MSG, W, H, BGCOLOR, SkString(#NAME)) |
| 53 | #define DEF_SIMPLE_GM_BG_NAME_CAN_FAIL(NAME, CANVAS, ERR_MSG, W, H, BGCOLOR, NAME_STR) \ |
| 54 | static skiagm::DrawResult SK_MACRO_CONCAT(NAME,_GM)(SkCanvas*, SkString*); \ |
| 55 | DEF_GM(return new skiagm::SimpleGM(BGCOLOR, NAME_STR, {W,H}, SK_MACRO_CONCAT(NAME,_GM));) \ |
| 56 | skiagm::DrawResult SK_MACRO_CONCAT(NAME,_GM)(SkCanvas* CANVAS, SkString* ERR_MSG) |
| 57 | |
halcanary | 30b83d4 | 2014-10-26 05:23:53 -0700 | [diff] [blame] | 58 | |
Chris Dalton | 3a77837 | 2019-02-07 15:23:36 -0700 | [diff] [blame] | 59 | // A Simple GpuGM makes direct GPU calls. Its onDraw hook that includes GPU objects as params, and |
| 60 | // is only invoked on GPU configs. Non-GPU configs automatically draw a GPU-only message and abort. |
| 61 | #define DEF_SIMPLE_GPU_GM(NAME, GR_CONTEXT, RENDER_TARGET_CONTEXT, CANVAS, W, H) \ |
| 62 | DEF_SIMPLE_GPU_GM_BG(NAME, GR_CONTEXT, RENDER_TARGET_CONTEXT, CANVAS, W, H, SK_ColorWHITE) |
Chris Dalton | 50e24d7 | 2019-02-07 16:20:09 -0700 | [diff] [blame] | 63 | #define DEF_SIMPLE_GPU_GM_BG(NAME, GR_CONTEXT, RENDER_TARGET_CONTEXT, CANVAS, W, H, BGCOLOR) \ |
| 64 | static void SK_MACRO_CONCAT(NAME,_GM_inner)(GrContext*, GrRenderTargetContext*, SkCanvas*); \ |
| 65 | DEF_SIMPLE_GPU_GM_BG_CAN_FAIL(NAME, GR_CONTEXT, RENDER_TARGET_CONTEXT, CANVAS,, W, H, \ |
| 66 | BGCOLOR) { \ |
| 67 | SK_MACRO_CONCAT(NAME,_GM_inner)(GR_CONTEXT, RENDER_TARGET_CONTEXT, CANVAS); \ |
| 68 | return skiagm::DrawResult::kOk; \ |
| 69 | } \ |
| 70 | void SK_MACRO_CONCAT(NAME,_GM_inner)( \ |
Chris Dalton | 3a77837 | 2019-02-07 15:23:36 -0700 | [diff] [blame] | 71 | GrContext* GR_CONTEXT, GrRenderTargetContext* RENDER_TARGET_CONTEXT, SkCanvas* CANVAS) |
| 72 | |
Chris Dalton | 50e24d7 | 2019-02-07 16:20:09 -0700 | [diff] [blame] | 73 | #define DEF_SIMPLE_GPU_GM_CAN_FAIL(NAME, GR_CONTEXT, RENDER_TARGET_CONTEXT, CANVAS, ERR_MSG, W, H) \ |
| 74 | DEF_SIMPLE_GPU_GM_BG_CAN_FAIL(NAME, GR_CONTEXT, RENDER_TARGET_CONTEXT, CANVAS, \ |
| 75 | ERR_MSG, W, H, SK_ColorWHITE) |
| 76 | #define DEF_SIMPLE_GPU_GM_BG_CAN_FAIL(NAME, GR_CONTEXT, RENDER_TARGET_CONTEXT, CANVAS, ERR_MSG, W, \ |
| 77 | H, BGCOLOR) \ |
| 78 | static skiagm::DrawResult SK_MACRO_CONCAT(NAME,_GM)( \ |
| 79 | GrContext*, GrRenderTargetContext*, SkCanvas*, SkString*); \ |
| 80 | DEF_GM(return new skiagm::SimpleGpuGM(BGCOLOR, SkString(#NAME), {W,H}, \ |
| 81 | SK_MACRO_CONCAT(NAME,_GM));) \ |
| 82 | skiagm::DrawResult SK_MACRO_CONCAT(NAME,_GM)( \ |
| 83 | GrContext* GR_CONTEXT, GrRenderTargetContext* RENDER_TARGET_CONTEXT, SkCanvas* CANVAS, \ |
| 84 | SkString* ERR_MSG) |
| 85 | |
reed@android.com | 00dae86 | 2009-06-10 15:38:48 +0000 | [diff] [blame] | 86 | namespace skiagm { |
rmistry@google.com | d6176b0 | 2012-08-23 18:14:13 +0000 | [diff] [blame] | 87 | |
Chris Dalton | 50e24d7 | 2019-02-07 16:20:09 -0700 | [diff] [blame] | 88 | enum class DrawResult { |
| 89 | kOk, // Test drew successfully. |
| 90 | kFail, // Test failed to draw. |
| 91 | kSkip // Test is not applicable in this context and should be skipped. |
| 92 | }; |
| 93 | |
reed@android.com | 00dae86 | 2009-06-10 15:38:48 +0000 | [diff] [blame] | 94 | class GM { |
| 95 | public: |
Chris Dalton | 50e24d7 | 2019-02-07 16:20:09 -0700 | [diff] [blame] | 96 | using DrawResult = skiagm::DrawResult; |
| 97 | |
Chris Dalton | 3a77837 | 2019-02-07 15:23:36 -0700 | [diff] [blame] | 98 | GM(SkColor backgroundColor = SK_ColorWHITE); |
reed@android.com | 00dae86 | 2009-06-10 15:38:48 +0000 | [diff] [blame] | 99 | virtual ~GM(); |
rmistry@google.com | d6176b0 | 2012-08-23 18:14:13 +0000 | [diff] [blame] | 100 | |
commit-bot@chromium.org | b21fac1 | 2014-02-07 21:13:11 +0000 | [diff] [blame] | 101 | enum Mode { |
| 102 | kGM_Mode, |
| 103 | kSample_Mode, |
| 104 | kBench_Mode, |
| 105 | }; |
| 106 | |
| 107 | void setMode(Mode mode) { fMode = mode; } |
| 108 | Mode getMode() const { return fMode; } |
| 109 | |
Chris Dalton | 50e24d7 | 2019-02-07 16:20:09 -0700 | [diff] [blame] | 110 | static constexpr char kErrorMsg_DrawSkippedGpuOnly[] = "This test is for GPU configs only."; |
| 111 | |
| 112 | DrawResult draw(SkCanvas* canvas) { |
| 113 | SkString errorMsg; |
| 114 | return this->draw(canvas, &errorMsg); |
| 115 | } |
| 116 | DrawResult draw(SkCanvas*, SkString* errorMsg); |
| 117 | |
bsalomon@google.com | 48dd1a2 | 2011-10-31 14:18:20 +0000 | [diff] [blame] | 118 | void drawBackground(SkCanvas*); |
Chris Dalton | 50e24d7 | 2019-02-07 16:20:09 -0700 | [diff] [blame] | 119 | DrawResult drawContent(SkCanvas* canvas) { |
| 120 | SkString errorMsg; |
| 121 | return this->drawContent(canvas, &errorMsg); |
| 122 | } |
| 123 | DrawResult drawContent(SkCanvas*, SkString* errorMsg); |
rmistry@google.com | d6176b0 | 2012-08-23 18:14:13 +0000 | [diff] [blame] | 124 | |
mtklein | f4ba321 | 2015-01-28 15:32:24 -0800 | [diff] [blame] | 125 | SkISize getISize() { return this->onISize(); } |
commit-bot@chromium.org | 38aeb0f | 2014-02-26 23:01:57 +0000 | [diff] [blame] | 126 | const char* getName(); |
reed@android.com | 00dae86 | 2009-06-10 15:38:48 +0000 | [diff] [blame] | 127 | |
Mike Klein | 9707e90 | 2019-02-07 16:18:22 -0500 | [diff] [blame] | 128 | virtual bool runAsBench() const; |
mtklein | cf5d9c9 | 2015-01-23 10:31:45 -0800 | [diff] [blame] | 129 | |
reed@google.com | 487b560 | 2013-02-01 15:01:24 +0000 | [diff] [blame] | 130 | SkScalar width() { |
| 131 | return SkIntToScalar(this->getISize().width()); |
| 132 | } |
| 133 | SkScalar height() { |
scroggo@google.com | ab02627 | 2013-04-12 22:14:03 +0000 | [diff] [blame] | 134 | return SkIntToScalar(this->getISize().height()); |
reed@google.com | 487b560 | 2013-02-01 15:01:24 +0000 | [diff] [blame] | 135 | } |
| 136 | |
reed@google.com | b4b49cc | 2011-12-06 16:15:42 +0000 | [diff] [blame] | 137 | SkColor getBGColor() const { return fBGColor; } |
bsalomon@google.com | 48dd1a2 | 2011-10-31 14:18:20 +0000 | [diff] [blame] | 138 | void setBGColor(SkColor); |
reed@google.com | 30db599 | 2011-08-29 17:41:02 +0000 | [diff] [blame] | 139 | |
Chris Dalton | 3a77837 | 2019-02-07 15:23:36 -0700 | [diff] [blame] | 140 | // helper: fill a rect in the specified color based on the GM's getISize bounds. |
reed@google.com | 2d6ef52 | 2012-01-03 17:20:38 +0000 | [diff] [blame] | 141 | void drawSizeBounds(SkCanvas*, SkColor); |
| 142 | |
reed@google.com | aef7361 | 2012-11-16 13:41:45 +0000 | [diff] [blame] | 143 | bool isCanvasDeferred() const { return fCanvasIsDeferred; } |
| 144 | void setCanvasIsDeferred(bool isDeferred) { |
| 145 | fCanvasIsDeferred = isDeferred; |
| 146 | } |
| 147 | |
Hal Canary | 4124807 | 2019-07-11 16:32:53 -0400 | [diff] [blame] | 148 | bool animate(double /*nanos*/); |
Hal Canary | c74a550 | 2019-07-08 14:55:15 -0400 | [diff] [blame] | 149 | virtual bool onChar(SkUnichar); |
reed | d9adfe6 | 2015-02-01 19:01:04 -0800 | [diff] [blame] | 150 | |
Mike Reed | 81f60ec | 2018-05-15 10:09:52 -0400 | [diff] [blame] | 151 | bool getControls(SkMetaData* controls) { return this->onGetControls(controls); } |
| 152 | void setControls(const SkMetaData& controls) { this->onSetControls(controls); } |
| 153 | |
Chris Dalton | 5b5403e | 2019-06-05 11:54:39 -0600 | [diff] [blame] | 154 | virtual void modifyGrContextOptions(GrContextOptions*); |
bsalomon | 4ee6bd8 | 2015-05-27 13:23:23 -0700 | [diff] [blame] | 155 | |
halcanary | 2a24338 | 2015-09-09 08:16:41 -0700 | [diff] [blame] | 156 | protected: |
Mike Klein | 9707e90 | 2019-02-07 16:18:22 -0500 | [diff] [blame] | 157 | virtual void onOnceBeforeDraw(); |
| 158 | virtual DrawResult onDraw(SkCanvas* canvas, SkString* errorMsg); |
| 159 | virtual void onDraw(SkCanvas*); |
| 160 | |
robertphillips@google.com | 8570b5c | 2012-03-20 17:40:58 +0000 | [diff] [blame] | 161 | virtual SkISize onISize() = 0; |
reed@android.com | 8015dd8 | 2009-06-21 00:49:18 +0000 | [diff] [blame] | 162 | virtual SkString onShortName() = 0; |
mtklein | 1c40292 | 2015-01-23 11:07:07 -0800 | [diff] [blame] | 163 | |
Hal Canary | 4124807 | 2019-07-11 16:32:53 -0400 | [diff] [blame] | 164 | virtual bool onAnimate(double /*nanos*/); |
Mike Klein | 9707e90 | 2019-02-07 16:18:22 -0500 | [diff] [blame] | 165 | virtual bool onGetControls(SkMetaData*); |
| 166 | virtual void onSetControls(const SkMetaData&); |
vandebo@chromium.org | 79d3cb4 | 2012-03-21 17:34:30 +0000 | [diff] [blame] | 167 | |
reed@android.com | 8015dd8 | 2009-06-21 00:49:18 +0000 | [diff] [blame] | 168 | private: |
commit-bot@chromium.org | b21fac1 | 2014-02-07 21:13:11 +0000 | [diff] [blame] | 169 | Mode fMode; |
reed@android.com | 8015dd8 | 2009-06-21 00:49:18 +0000 | [diff] [blame] | 170 | SkString fShortName; |
bsalomon@google.com | 48dd1a2 | 2011-10-31 14:18:20 +0000 | [diff] [blame] | 171 | SkColor fBGColor; |
reed@google.com | aef7361 | 2012-11-16 13:41:45 +0000 | [diff] [blame] | 172 | bool fCanvasIsDeferred; // work-around problem in srcmode.cpp |
reed@google.com | 7775d66 | 2012-11-27 15:15:58 +0000 | [diff] [blame] | 173 | bool fHaveCalledOnceBeforeDraw; |
reed@android.com | 00dae86 | 2009-06-10 15:38:48 +0000 | [diff] [blame] | 174 | }; |
| 175 | |
Hal Canary | edda565 | 2019-08-05 10:28:09 -0400 | [diff] [blame] | 176 | using GMFactory = std::unique_ptr<skiagm::GM> (*)(); |
| 177 | using GMRegistry = sk_tools::Registry<GMFactory>; |
halcanary | f62c634 | 2015-01-12 15:27:46 -0800 | [diff] [blame] | 178 | |
Chris Dalton | 3a77837 | 2019-02-07 15:23:36 -0700 | [diff] [blame] | 179 | // A GpuGM replaces the onDraw method with one that also accepts GPU objects alongside the |
| 180 | // SkCanvas. Its onDraw is only invoked on GPU configs; on non-GPU configs it will automatically |
| 181 | // draw a GPU-only message and abort. |
| 182 | class GpuGM : public GM { |
halcanary | f62c634 | 2015-01-12 15:27:46 -0800 | [diff] [blame] | 183 | public: |
Chris Dalton | 3a77837 | 2019-02-07 15:23:36 -0700 | [diff] [blame] | 184 | GpuGM(SkColor backgroundColor = SK_ColorWHITE) : GM(backgroundColor) {} |
Chris Dalton | 382b122 | 2019-02-07 10:05:55 +0000 | [diff] [blame] | 185 | private: |
Chris Dalton | 50e24d7 | 2019-02-07 16:20:09 -0700 | [diff] [blame] | 186 | using GM::onDraw; |
| 187 | DrawResult onDraw(SkCanvas*, SkString* errorMsg) final; |
Mike Klein | 9707e90 | 2019-02-07 16:18:22 -0500 | [diff] [blame] | 188 | |
Chris Dalton | 50e24d7 | 2019-02-07 16:20:09 -0700 | [diff] [blame] | 189 | virtual DrawResult onDraw(GrContext* ctx, GrRenderTargetContext* rtc, SkCanvas* canvas, |
Mike Klein | 9707e90 | 2019-02-07 16:18:22 -0500 | [diff] [blame] | 190 | SkString* errorMsg); |
| 191 | virtual void onDraw(GrContext*, GrRenderTargetContext*, SkCanvas*); |
Chris Dalton | f5efa78 | 2019-02-05 17:42:14 -0700 | [diff] [blame] | 192 | }; |
Chris Dalton | 3a77837 | 2019-02-07 15:23:36 -0700 | [diff] [blame] | 193 | |
| 194 | // SimpleGM is intended for basic GMs that can define their entire implementation inside a |
| 195 | // single "draw" function pointer. |
| 196 | class SimpleGM : public GM { |
| 197 | public: |
Chris Dalton | 50e24d7 | 2019-02-07 16:20:09 -0700 | [diff] [blame] | 198 | using DrawProc = DrawResult(*)(SkCanvas*, SkString*); |
Chris Dalton | 3a77837 | 2019-02-07 15:23:36 -0700 | [diff] [blame] | 199 | SimpleGM(SkColor bgColor, const SkString& name, const SkISize& size, DrawProc drawProc) |
| 200 | : GM(bgColor), fName(name), fSize(size), fDrawProc(drawProc) {} |
| 201 | |
| 202 | private: |
Mike Klein | 9707e90 | 2019-02-07 16:18:22 -0500 | [diff] [blame] | 203 | SkISize onISize() override; |
| 204 | SkString onShortName() override; |
| 205 | DrawResult onDraw(SkCanvas* canvas, SkString* errorMsg) override; |
Chris Dalton | 3a77837 | 2019-02-07 15:23:36 -0700 | [diff] [blame] | 206 | |
| 207 | const SkString fName; |
| 208 | const SkISize fSize; |
| 209 | const DrawProc fDrawProc; |
| 210 | }; |
| 211 | |
| 212 | class SimpleGpuGM : public GpuGM { |
| 213 | public: |
Chris Dalton | 50e24d7 | 2019-02-07 16:20:09 -0700 | [diff] [blame] | 214 | using DrawProc = DrawResult(*)(GrContext*, GrRenderTargetContext*, SkCanvas*, SkString*); |
Chris Dalton | 3a77837 | 2019-02-07 15:23:36 -0700 | [diff] [blame] | 215 | SimpleGpuGM(SkColor bgColor, const SkString& name, const SkISize& size, DrawProc drawProc) |
| 216 | : GpuGM(bgColor), fName(name), fSize(size), fDrawProc(drawProc) {} |
| 217 | |
| 218 | private: |
Mike Klein | 9707e90 | 2019-02-07 16:18:22 -0500 | [diff] [blame] | 219 | SkISize onISize() override; |
| 220 | SkString onShortName() override; |
Chris Dalton | 50e24d7 | 2019-02-07 16:20:09 -0700 | [diff] [blame] | 221 | DrawResult onDraw(GrContext* ctx, GrRenderTargetContext* rtc, SkCanvas* canvas, |
Mike Klein | 9707e90 | 2019-02-07 16:18:22 -0500 | [diff] [blame] | 222 | SkString* errorMsg) override; |
Chris Dalton | 3a77837 | 2019-02-07 15:23:36 -0700 | [diff] [blame] | 223 | |
| 224 | const SkString fName; |
| 225 | const SkISize fSize; |
| 226 | const DrawProc fDrawProc; |
| 227 | }; |
reed@android.com | 00dae86 | 2009-06-10 15:38:48 +0000 | [diff] [blame] | 228 | } |
| 229 | |
Mike Klein | c9eace8 | 2018-10-31 10:49:38 -0400 | [diff] [blame] | 230 | void MarkGMGood(SkCanvas*, SkScalar x, SkScalar y); |
| 231 | void MarkGMBad (SkCanvas*, SkScalar x, SkScalar y); |
| 232 | |
reed@android.com | 00dae86 | 2009-06-10 15:38:48 +0000 | [diff] [blame] | 233 | #endif |