| 1 | unit BGRAGrayscaleMask;
|
|---|
| 2 |
|
|---|
| 3 | {$mode objfpc}{$H+}
|
|---|
| 4 |
|
|---|
| 5 | interface
|
|---|
| 6 |
|
|---|
| 7 | uses
|
|---|
| 8 | Classes, SysUtils, BGRABitmapTypes;
|
|---|
| 9 |
|
|---|
| 10 | type
|
|---|
| 11 | { TGrayscaleMask }
|
|---|
| 12 |
|
|---|
| 13 | TGrayscaleMask = class
|
|---|
| 14 | private
|
|---|
| 15 | FData: PByte;
|
|---|
| 16 | FWidth, FHeight, FNbPixels: Integer;
|
|---|
| 17 | function GetScanLine(Y: Integer): PByte;
|
|---|
| 18 | procedure Init(AWidth,AHeight: Integer);
|
|---|
| 19 | public
|
|---|
| 20 | constructor Create(AWidth,AHeight: Integer); overload;
|
|---|
| 21 | constructor Create(AWidth,AHeight: Integer; AValue: byte); overload;
|
|---|
| 22 | constructor Create(ABitmap: TBGRACustomBitmap; AChannel: TChannel); overload;
|
|---|
| 23 | constructor CreateDownSample(ABitmap: TBGRACustomBitmap; AWidth,AHeight: integer);
|
|---|
| 24 | procedure CopyFrom(ABitmap: TBGRACustomBitmap; AChannel: TChannel);
|
|---|
| 25 | procedure Draw(ABitmap: TBGRACustomBitmap; X,Y: Integer);
|
|---|
| 26 | procedure DrawAsAlpha(ABitmap: TBGRACustomBitmap; X,Y: Integer; const c: TBGRAPixel); overload;
|
|---|
| 27 | procedure DrawAsAlpha(ABitmap: TBGRACustomBitmap; X,Y: Integer; texture: IBGRAScanner); overload;
|
|---|
| 28 | destructor Destroy; override;
|
|---|
| 29 | function GetPixel(X,Y: integer): byte;
|
|---|
| 30 | procedure SetPixel(X,Y: integer; AValue: byte);
|
|---|
| 31 | property ScanLine[Y: Integer]: PByte read GetScanLine;
|
|---|
| 32 | property Data: PByte read FData;
|
|---|
| 33 | property Width: Integer read FWidth;
|
|---|
| 34 | property Height: Integer read FHeight;
|
|---|
| 35 | property NbPixels: Integer read FNbPixels;
|
|---|
| 36 | end;
|
|---|
| 37 |
|
|---|
| 38 | procedure DownSamplePutImageGrayscale(sourceData: PByte; sourcePixelSize: NativeInt; sourceRowDelta: NativeInt; sourceWidth, sourceHeight: NativeInt; dest: TGrayscaleMask; ADestRect: TRect); overload;
|
|---|
| 39 | procedure DownSamplePutImageGrayscale(source: TBGRACustomBitmap; dest: TGrayscaleMask; ADestRect: TRect); overload;
|
|---|
| 40 |
|
|---|
| 41 | procedure BGRAFillClearTypeGrayscaleMask(dest: TBGRACustomBitmap; x,
|
|---|
| 42 | y: integer; xThird: integer; mask: TGrayscaleMask; color: TBGRAPixel;
|
|---|
| 43 | texture: IBGRAScanner; RGBOrder: boolean);
|
|---|
| 44 |
|
|---|
| 45 | implementation
|
|---|
| 46 |
|
|---|
| 47 | uses BGRABlend;
|
|---|
| 48 |
|
|---|
| 49 | procedure BGRAFillClearTypeGrayscaleMask(dest: TBGRACustomBitmap; x,
|
|---|
| 50 | y: integer; xThird: integer; mask: TGrayscaleMask; color: TBGRAPixel;
|
|---|
| 51 | texture: IBGRAScanner; RGBOrder: boolean);
|
|---|
| 52 | var delta: NativeInt;
|
|---|
| 53 | begin
|
|---|
| 54 | delta := mask.Width;
|
|---|
| 55 | BGRABlend.BGRAFillClearTypeMaskPtr(dest,x,y,xThird,mask.ScanLine[0],1,delta,mask.Width,mask.Height,color,texture,RGBOrder);
|
|---|
| 56 | end;
|
|---|
| 57 |
|
|---|
| 58 | { TGrayscaleMask }
|
|---|
| 59 |
|
|---|
| 60 | function TGrayscaleMask.GetScanLine(Y: Integer): PByte;
|
|---|
| 61 | begin
|
|---|
| 62 | if (y < 0) or (y >= FHeight) then
|
|---|
| 63 | raise ERangeError.Create('Scanline: out of bounds');
|
|---|
| 64 | result := FData + NativeInt(Y)*NativeInt(FWidth);
|
|---|
| 65 | end;
|
|---|
| 66 |
|
|---|
| 67 | procedure TGrayscaleMask.Init(AWidth, AHeight: Integer);
|
|---|
| 68 | begin
|
|---|
| 69 | if FData <> nil then
|
|---|
| 70 | begin
|
|---|
| 71 | FreeMem(FData);
|
|---|
| 72 | FData := nil;
|
|---|
| 73 | end;
|
|---|
| 74 | FWidth := AWidth; if FWidth < 0 then FWidth:= 0;
|
|---|
| 75 | FHeight := AHeight; if FHeight < 0 then FHeight:= 0;
|
|---|
| 76 | FNbPixels:= FWidth*FHeight;
|
|---|
| 77 | if FNbPixels > 0 then getmem(FData, FNbPixels);
|
|---|
| 78 | end;
|
|---|
| 79 |
|
|---|
| 80 | procedure TGrayscaleMask.CopyFrom(ABitmap: TBGRACustomBitmap; AChannel: TChannel
|
|---|
| 81 | );
|
|---|
| 82 | var psrc: PByte;
|
|---|
| 83 | pdest: PByte;
|
|---|
| 84 | x,y: integer;
|
|---|
| 85 | ofs: NativeInt;
|
|---|
| 86 | begin
|
|---|
| 87 | Init(ABitmap.Width,ABitmap.Height);
|
|---|
| 88 | if FNbPixels > 0 then
|
|---|
| 89 | begin
|
|---|
| 90 | pdest := FData;
|
|---|
| 91 | Case AChannel of
|
|---|
| 92 | cAlpha: ofs := TBGRAPixel_AlphaByteOffset;
|
|---|
| 93 | cRed: ofs := TBGRAPixel_RedByteOffset;
|
|---|
| 94 | cGreen: ofs := TBGRAPixel_GreenByteOffset;
|
|---|
| 95 | else
|
|---|
| 96 | ofs := TBGRAPixel_BlueByteOffset;
|
|---|
| 97 | end;
|
|---|
| 98 | for y := 0 to FHeight-1 do
|
|---|
| 99 | begin
|
|---|
| 100 | psrc := PByte(ABitmap.ScanLine[y])+ofs;
|
|---|
| 101 | for x := FWidth-1 downto 0 do
|
|---|
| 102 | begin
|
|---|
| 103 | pdest^ := psrc^;
|
|---|
| 104 | inc(pdest);
|
|---|
| 105 | inc(psrc,sizeof(TBGRAPixel));
|
|---|
| 106 | end;
|
|---|
| 107 | end;
|
|---|
| 108 | end;
|
|---|
| 109 | end;
|
|---|
| 110 |
|
|---|
| 111 | constructor TGrayscaleMask.Create(AWidth, AHeight: Integer);
|
|---|
| 112 | begin
|
|---|
| 113 | Init(AWidth,AHeight);
|
|---|
| 114 | if FNbPixels > 0 then FillChar(FData^, FNbPixels, 0);
|
|---|
| 115 | end;
|
|---|
| 116 |
|
|---|
| 117 | constructor TGrayscaleMask.Create(AWidth, AHeight: Integer; AValue: byte);
|
|---|
| 118 | begin
|
|---|
| 119 | Init(AWidth,AHeight);
|
|---|
| 120 | if FNbPixels > 0 then FillChar(FData^, FNbPixels, AValue);
|
|---|
| 121 | end;
|
|---|
| 122 |
|
|---|
| 123 | constructor TGrayscaleMask.Create(ABitmap: TBGRACustomBitmap; AChannel: TChannel);
|
|---|
| 124 | begin
|
|---|
| 125 | CopyFrom(ABitmap, AChannel);
|
|---|
| 126 | end;
|
|---|
| 127 |
|
|---|
| 128 | constructor TGrayscaleMask.CreateDownSample(ABitmap: TBGRACustomBitmap; AWidth,
|
|---|
| 129 | AHeight: integer);
|
|---|
| 130 | begin
|
|---|
| 131 | if (AWidth = ABitmap.Width) and (AHeight = ABitmap.Height) then
|
|---|
| 132 | CopyFrom(ABitmap,cGreen)
|
|---|
| 133 | else
|
|---|
| 134 | begin
|
|---|
| 135 | if (ABitmap.Width < AWidth) or (ABitmap.Height < AHeight) then
|
|---|
| 136 | raise exception.Create('Original size smaller');
|
|---|
| 137 | Init(AWidth,AHeight);
|
|---|
| 138 | if FNbPixels > 0 then
|
|---|
| 139 | DownSamplePutImageGrayscale(ABitmap, self, rect(0,0,FWidth,FHeight));
|
|---|
| 140 | end;
|
|---|
| 141 | end;
|
|---|
| 142 |
|
|---|
| 143 | procedure TGrayscaleMask.Draw(ABitmap: TBGRACustomBitmap; X, Y: Integer);
|
|---|
| 144 | var
|
|---|
| 145 | yb, minxb, minyb, maxxb, maxyb, ignoreleft, copycount,
|
|---|
| 146 | i, delta_source, delta_dest: integer;
|
|---|
| 147 | pdest: PBGRAPixel;
|
|---|
| 148 | psource: PByte;
|
|---|
| 149 | value: byte;
|
|---|
| 150 | begin
|
|---|
| 151 | if not CheckPutImageBounds(x,y,FWidth,Fheight,minxb,minyb,maxxb,maxyb,ignoreleft,ABitmap.ClipRect) then exit;
|
|---|
| 152 | copycount := maxxb - minxb + 1;
|
|---|
| 153 |
|
|---|
| 154 | psource := ScanLine[minyb - y] + ignoreleft;
|
|---|
| 155 | delta_source := FWidth;
|
|---|
| 156 |
|
|---|
| 157 | pdest := ABitmap.Scanline[minyb] + minxb;
|
|---|
| 158 | if ABitmap.LineOrder = riloBottomToTop then
|
|---|
| 159 | delta_dest := -ABitmap.Width
|
|---|
| 160 | else
|
|---|
| 161 | delta_dest := ABitmap.Width;
|
|---|
| 162 |
|
|---|
| 163 | Dec(delta_source, copycount);
|
|---|
| 164 | Dec(delta_dest, copycount);
|
|---|
| 165 | for yb := minyb to maxyb do
|
|---|
| 166 | begin
|
|---|
| 167 | for i := copycount -1 downto 0 do
|
|---|
| 168 | begin
|
|---|
| 169 | value := psource^;
|
|---|
| 170 | pdest^ := BGRA(value,value,value,255);
|
|---|
| 171 | inc(psource);
|
|---|
| 172 | inc(pdest);
|
|---|
| 173 | end;
|
|---|
| 174 | Inc(psource, delta_source);
|
|---|
| 175 | Inc(pdest, delta_dest);
|
|---|
| 176 | end;
|
|---|
| 177 | ABitmap.InvalidateBitmap;
|
|---|
| 178 | end;
|
|---|
| 179 |
|
|---|
| 180 | procedure TGrayscaleMask.DrawAsAlpha(ABitmap: TBGRACustomBitmap; X, Y: Integer;
|
|---|
| 181 | const c: TBGRAPixel);
|
|---|
| 182 | var
|
|---|
| 183 | yb, minxb, minyb, maxxb, maxyb, ignoreleft, copycount,
|
|---|
| 184 | i, delta_source, delta_dest: integer;
|
|---|
| 185 | pdest: PBGRAPixel;
|
|---|
| 186 | psource: PByte;
|
|---|
| 187 | begin
|
|---|
| 188 | if not CheckPutImageBounds(x,y,FWidth,Fheight,minxb,minyb,maxxb,maxyb,ignoreleft,ABitmap.ClipRect) then exit;
|
|---|
| 189 | copycount := maxxb - minxb + 1;
|
|---|
| 190 |
|
|---|
| 191 | psource := ScanLine[minyb - y] + ignoreleft;
|
|---|
| 192 | delta_source := FWidth;
|
|---|
| 193 |
|
|---|
| 194 | pdest := ABitmap.Scanline[minyb] + minxb;
|
|---|
| 195 | if ABitmap.LineOrder = riloBottomToTop then
|
|---|
| 196 | delta_dest := -ABitmap.Width
|
|---|
| 197 | else
|
|---|
| 198 | delta_dest := ABitmap.Width;
|
|---|
| 199 |
|
|---|
| 200 | Dec(delta_source, copycount);
|
|---|
| 201 | Dec(delta_dest, copycount);
|
|---|
| 202 | for yb := minyb to maxyb do
|
|---|
| 203 | begin
|
|---|
| 204 | for i := copycount -1 downto 0 do
|
|---|
| 205 | begin
|
|---|
| 206 | DrawPixelInlineWithAlphaCheck(pdest,c,psource^);
|
|---|
| 207 | inc(psource);
|
|---|
| 208 | inc(pdest);
|
|---|
| 209 | end;
|
|---|
| 210 | Inc(psource, delta_source);
|
|---|
| 211 | Inc(pdest, delta_dest);
|
|---|
| 212 | end;
|
|---|
| 213 | ABitmap.InvalidateBitmap;
|
|---|
| 214 | end;
|
|---|
| 215 |
|
|---|
| 216 | procedure TGrayscaleMask.DrawAsAlpha(ABitmap: TBGRACustomBitmap; X, Y: Integer;
|
|---|
| 217 | texture: IBGRAScanner);
|
|---|
| 218 | var
|
|---|
| 219 | yb, minxb, minyb, maxxb, maxyb, ignoreleft, copycount,
|
|---|
| 220 | i, delta_source, delta_dest: integer;
|
|---|
| 221 | pdest,ptex: PBGRAPixel;
|
|---|
| 222 | psource: PByte;
|
|---|
| 223 | memScan: PBGRAPixel;
|
|---|
| 224 | begin
|
|---|
| 225 | if not CheckPutImageBounds(x,y,FWidth,Fheight,minxb,minyb,maxxb,maxyb,ignoreleft,ABitmap.ClipRect) then exit;
|
|---|
| 226 | copycount := maxxb - minxb + 1;
|
|---|
| 227 | if copycount <= 0 then exit;
|
|---|
| 228 |
|
|---|
| 229 | psource := ScanLine[minyb - y] + ignoreleft;
|
|---|
| 230 | delta_source := FWidth;
|
|---|
| 231 |
|
|---|
| 232 | pdest := ABitmap.Scanline[minyb] + minxb;
|
|---|
| 233 | if ABitmap.LineOrder = riloBottomToTop then
|
|---|
| 234 | delta_dest := -ABitmap.Width
|
|---|
| 235 | else
|
|---|
| 236 | delta_dest := ABitmap.Width;
|
|---|
| 237 |
|
|---|
| 238 | getmem(memscan, copycount*sizeof(TBGRAPixel));
|
|---|
| 239 |
|
|---|
| 240 | Dec(delta_source, copycount);
|
|---|
| 241 | Dec(delta_dest, copycount);
|
|---|
| 242 | for yb := minyb to maxyb do
|
|---|
| 243 | begin
|
|---|
| 244 | texture.ScanMoveTo(ignoreleft,yb-y);
|
|---|
| 245 | texture.ScanPutPixels(memscan,copycount,dmSet);
|
|---|
| 246 | ptex := memScan;
|
|---|
| 247 | for i := copycount -1 downto 0 do
|
|---|
| 248 | begin
|
|---|
| 249 | DrawPixelInlineWithAlphaCheck(pdest,ptex^,psource^);
|
|---|
| 250 | inc(psource);
|
|---|
| 251 | inc(pdest);
|
|---|
| 252 | inc(ptex);
|
|---|
| 253 | end;
|
|---|
| 254 | Inc(psource, delta_source);
|
|---|
| 255 | Inc(pdest, delta_dest);
|
|---|
| 256 | end;
|
|---|
| 257 | ABitmap.InvalidateBitmap;
|
|---|
| 258 | freemem(memscan);
|
|---|
| 259 | end;
|
|---|
| 260 |
|
|---|
| 261 | destructor TGrayscaleMask.Destroy;
|
|---|
| 262 | begin
|
|---|
| 263 | if FData <> nil then
|
|---|
| 264 | begin
|
|---|
| 265 | freemem(FData);
|
|---|
| 266 | FData := nil;
|
|---|
| 267 | end;
|
|---|
| 268 | inherited Destroy;
|
|---|
| 269 | end;
|
|---|
| 270 |
|
|---|
| 271 | function TGrayscaleMask.GetPixel(X, Y: integer): byte;
|
|---|
| 272 | begin
|
|---|
| 273 | if (x < 0) or (x >= FWidth) then
|
|---|
| 274 | raise ERangeError.Create('GetPixel: out of bounds');
|
|---|
| 275 | result := (ScanLine[Y]+X)^;
|
|---|
| 276 | end;
|
|---|
| 277 |
|
|---|
| 278 | procedure TGrayscaleMask.SetPixel(X, Y: integer; AValue: byte);
|
|---|
| 279 | begin
|
|---|
| 280 | if (x < 0) or (x >= FWidth) then
|
|---|
| 281 | raise ERangeError.Create('SetPixel: out of bounds');
|
|---|
| 282 | (ScanLine[Y]+X)^ := AValue;
|
|---|
| 283 | end;
|
|---|
| 284 |
|
|---|
| 285 | procedure DownSamplePutImageGrayscale(sourceData: PByte;
|
|---|
| 286 | sourcePixelSize: NativeInt; sourceRowDelta: NativeInt; sourceWidth,
|
|---|
| 287 | sourceHeight: NativeInt; dest: TGrayscaleMask; ADestRect: TRect);
|
|---|
| 288 | var
|
|---|
| 289 | x_dest,y_dest: integer;
|
|---|
| 290 | pdest: PByte;
|
|---|
| 291 | nbPix,sum: NativeUInt;
|
|---|
| 292 | prev_x_src,x_src,x_src_nb,xb: NativeInt;
|
|---|
| 293 | x_src_inc,x_src_acc,x_src_div,x_src_rest: NativeInt;
|
|---|
| 294 | prev_y_src,y_src,y_src_nb,yb: NativeInt;
|
|---|
| 295 | y_src_inc,y_src_acc,y_src_div,y_src_rest: NativeInt;
|
|---|
| 296 | psrc,psrc2,psrc3: PByte;
|
|---|
| 297 | begin
|
|---|
| 298 | y_src_div := ADestRect.Bottom-ADestRect.Top;
|
|---|
| 299 | y_src_inc := sourceHeight div y_src_div;
|
|---|
| 300 | y_src_rest := sourceHeight mod y_src_div;
|
|---|
| 301 | x_src_div := ADestRect.Right-ADestRect.Left;
|
|---|
| 302 | x_src_inc := sourceWidth div x_src_div;
|
|---|
| 303 | x_src_rest := sourceWidth mod x_src_div;
|
|---|
| 304 |
|
|---|
| 305 | if (x_src_rest = 0) and (y_src_rest = 0) then
|
|---|
| 306 | begin
|
|---|
| 307 | x_src_nb := x_src_inc;
|
|---|
| 308 | y_src_nb := y_src_inc;
|
|---|
| 309 | nbPix := x_src_nb*y_src_nb;
|
|---|
| 310 | y_src := 0;
|
|---|
| 311 | for y_dest := ADestRect.Top to ADestRect.Bottom-1 do
|
|---|
| 312 | begin
|
|---|
| 313 | pdest := dest.ScanLine[y_dest]+ADestRect.Left;
|
|---|
| 314 | psrc := sourceData + y_src*sourceRowDelta;
|
|---|
| 315 | inc(y_src,y_src_inc);
|
|---|
| 316 |
|
|---|
| 317 | for x_dest := ADestRect.Right-ADestRect.Left-1 downto 0 do
|
|---|
| 318 | begin
|
|---|
| 319 | sum := 0;
|
|---|
| 320 | psrc2 := psrc;
|
|---|
| 321 | for xb := x_src_nb-1 downto 0 do
|
|---|
| 322 | begin
|
|---|
| 323 | psrc3 := psrc2;
|
|---|
| 324 | for yb := y_src_nb-1 downto 0 do
|
|---|
| 325 | begin
|
|---|
| 326 | inc(sum, psrc3^);
|
|---|
| 327 | inc(psrc3, sourceRowDelta);
|
|---|
| 328 | end;
|
|---|
| 329 | inc(psrc2, sourcePixelSize);
|
|---|
| 330 | end;
|
|---|
| 331 | pdest^ := sum div nbPix;
|
|---|
| 332 |
|
|---|
| 333 | psrc := psrc2;
|
|---|
| 334 | inc(pdest);
|
|---|
| 335 | end;
|
|---|
| 336 | end;
|
|---|
| 337 | end else
|
|---|
| 338 | begin
|
|---|
| 339 | y_src := 0;
|
|---|
| 340 | y_src_acc := 0;
|
|---|
| 341 | for y_dest := ADestRect.Top to ADestRect.Bottom-1 do
|
|---|
| 342 | begin
|
|---|
| 343 | pdest := dest.ScanLine[y_dest]+ADestRect.Left;
|
|---|
| 344 | psrc := sourceData + y_src*sourceRowDelta;
|
|---|
| 345 |
|
|---|
| 346 | prev_y_src := y_src;
|
|---|
| 347 | inc(y_src,y_src_inc);
|
|---|
| 348 | inc(y_src_acc,y_src_rest);
|
|---|
| 349 | if y_src_acc >= y_src_div then
|
|---|
| 350 | begin
|
|---|
| 351 | dec(y_src_acc,y_src_div);
|
|---|
| 352 | inc(y_src);
|
|---|
| 353 | end;
|
|---|
| 354 | y_src_nb := y_src-prev_y_src;
|
|---|
| 355 |
|
|---|
| 356 | x_src := 0;
|
|---|
| 357 | x_src_acc := 0;
|
|---|
| 358 | for x_dest := ADestRect.Right-ADestRect.Left-1 downto 0 do
|
|---|
| 359 | begin
|
|---|
| 360 | prev_x_src := x_src;
|
|---|
| 361 | inc(x_src,x_src_inc);
|
|---|
| 362 | inc(x_src_acc,x_src_rest);
|
|---|
| 363 | if x_src_acc >= x_src_div then
|
|---|
| 364 | begin
|
|---|
| 365 | dec(x_src_acc,x_src_div);
|
|---|
| 366 | inc(x_src);
|
|---|
| 367 | end;
|
|---|
| 368 | x_src_nb := x_src-prev_x_src;
|
|---|
| 369 |
|
|---|
| 370 | sum := 0;
|
|---|
| 371 | nbPix := 0;
|
|---|
| 372 | psrc2 := psrc;
|
|---|
| 373 | for xb := x_src_nb-1 downto 0 do
|
|---|
| 374 | begin
|
|---|
| 375 | psrc3 := psrc2;
|
|---|
| 376 | for yb := y_src_nb-1 downto 0 do
|
|---|
| 377 | begin
|
|---|
| 378 | inc(nbPix);
|
|---|
| 379 | inc(sum, psrc3^);
|
|---|
| 380 | inc(psrc3, sourceRowDelta);
|
|---|
| 381 | end;
|
|---|
| 382 | inc(psrc2, sourcePixelSize);
|
|---|
| 383 | end;
|
|---|
| 384 | pdest^ := sum div nbPix;
|
|---|
| 385 |
|
|---|
| 386 | psrc := psrc2;
|
|---|
| 387 | inc(pdest);
|
|---|
| 388 | end;
|
|---|
| 389 | end;
|
|---|
| 390 | end;
|
|---|
| 391 | end;
|
|---|
| 392 |
|
|---|
| 393 | procedure DownSamplePutImageGrayscale(source: TBGRACustomBitmap;
|
|---|
| 394 | dest: TGrayscaleMask; ADestRect: TRect);
|
|---|
| 395 | var delta: NativeInt;
|
|---|
| 396 | begin
|
|---|
| 397 | delta := source.Width*sizeof(TBGRAPixel);
|
|---|
| 398 | if source.LineOrder = riloBottomToTop then
|
|---|
| 399 | delta := -delta;
|
|---|
| 400 | DownSamplePutImageGrayscale(PByte(source.ScanLine[0])+1,sizeof(TBGRAPixel),delta,source.Width,source.Height,dest,ADestRect);
|
|---|
| 401 | end;
|
|---|
| 402 |
|
|---|
| 403 | end.
|
|---|
| 404 |
|
|---|