{ Solutions to the Advent Of Code. Copyright (C) 2023 Stefan Müller This program is free software: you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program. If not, see . } unit USandSlabs; {$mode ObjFPC}{$H+} interface uses Classes, SysUtils, Generics.Collections, Generics.Defaults, USolver; const CGroundSize = 10; type TBrick = class; TBricks = specialize TObjectList; { TBrick } TBrick = class private FX1, FY1, FZ1, FX2, FY2, FZ2: Integer; FIsDisintegratable: Boolean; FSupportBricks, FTopBricks: TBricks; public property X1: Integer read FX1; property Y1: Integer read FY1; property Z1: Integer read FZ1; property X2: Integer read FX2; property Y2: Integer read FY2; property Z2: Integer read FZ2; property IsDisintegratable: Boolean read FIsDisintegratable write FIsDisintegratable; property SupportBricks: TBricks read FSupportBricks; constructor Create(const AX1, AY1, AZ1, AX2, AY2, AZ2: Integer); constructor Create(const ALine: string); destructor Destroy; override; procedure SetZ1(const AValue: Integer); procedure AddTopBrick(const ABrick: TBrick); function CalcChainCount: Integer; end; { TBrickComparer } TBrickComparer = class(TInterfacedObject, specialize IComparer) public function Compare(constref ALeft, ARight: TBrick): Integer; overload; end; { TGroundTile } TGroundTile = record Height: Integer; TopBrick: TBrick; end; TGround = array[0..CGroundSize - 1, 0.. CGroundSize - 1] of TGroundTile; { TSandSlabs } TSandSlabs = class(TSolver) private FBricks: TBricks; FGround: TGround; procedure InitGround; procedure StackBrick(const ABrick: TBrick); public constructor Create; destructor Destroy; override; procedure ProcessDataLine(const ALine: string); override; procedure Finish; override; function GetDataFileName: string; override; function GetPuzzleName: string; override; end; implementation { TBrick } constructor TBrick.Create(const AX1, AY1, AZ1, AX2, AY2, AZ2: Integer); begin FX1 := AX1; FY1 := AY1; FZ1 := AZ1; FX2 := AX2; FY2 := AY2; FZ2 := AZ2; FIsDisintegratable := True; FSupportBricks := TBricks.Create(False); FTopBricks := TBricks.Create(False); end; constructor TBrick.Create(const ALine: string); var split: TStringArray; begin split := ALine.Split([',', '~']); Create(StrToInt(split[0]), StrToInt(split[1]), StrToInt(split[2]), StrToInt(split[3]), StrToInt(split[4]), StrToInt(split[5])); end; destructor TBrick.Destroy; begin FSupportBricks.Free; FTopBricks.Free; inherited Destroy; end; procedure TBrick.SetZ1(const AValue: Integer); begin Inc(FZ2, AValue - FZ1); FZ1 := AValue; end; procedure TBrick.AddTopBrick(const ABrick: TBrick); begin FTopBricks.Add(ABrick); end; function TBrick.CalcChainCount: Integer; var chainBricks, uncheckedTopBricks: TBricks; support, chain: TBrick; allSupportsInChain, foundInChain: Boolean; begin if FIsDisintegratable then Result := 0 else begin chainBricks := TBricks.Create(False); uncheckedTopBricks := TBricks.Create(TBrickComparer.Create, False); uncheckedTopBricks.AddRange(FTopBricks); while uncheckedTopBricks.Count > 0 do begin uncheckedTopBricks.Sort; foundInChain := False; for chain in chainBricks do if chain = uncheckedTopBricks[0] then foundInChain := True; if not foundInChain then begin allSupportsInChain := True; for support in uncheckedTopBricks[0].FSupportBricks do if support <> Self then begin foundInChain := False; for chain in chainBricks do if chain = support then begin foundInChain := True; Break; end; if not foundInChain then begin allSupportsInChain := False; Break; end; end; if allSupportsInChain then begin chainBricks.Add(uncheckedTopBricks[0]); uncheckedTopBricks.AddRange(uncheckedTopBricks[0].FTopBricks); end; end; uncheckedTopBricks.Delete(0); end; uncheckedTopBricks.Free; Result := chainBricks.Count; chainBricks.Free; end; end; { TBrickComparer } function TBrickComparer.Compare(constref ALeft, ARight: TBrick): Integer; begin Result := ALeft.FZ1 - ARight.FZ1; end; { TSandSlabs } procedure TSandSlabs.InitGround; var i, j: Integer; begin for i := 0 to CGroundSize - 1 do for j := 0 to CGroundSize - 1 do begin FGround[i, j].Height := 0; FGround[i, j].TopBrick := nil; end; end; procedure TSandSlabs.StackBrick(const ABrick: TBrick); var i, j, max: Integer; begin Inc(FPart1); // Checks height and supportBricks for this brick. max := 0; for i := ABrick.X1 to ABrick.X2 do for j := ABrick.Y1 to ABrick.Y2 do if max <= FGround[i, j].Height then begin if max < FGround[i, j].Height then begin max := FGround[i, j].Height; ABrick.SupportBricks.Clear; end; if (FGround[i, j].TopBrick <> nil) and not ABrick.SupportBricks.Contains(FGround[i, j].TopBrick) then ABrick.SupportBricks.Add(FGround[i, j].TopBrick); end; // Updates disintegration flag. if ABrick.SupportBricks.Count = 1 then begin if ABrick.SupportBricks[0].IsDisintegratable then begin ABrick.SupportBricks[0].IsDisintegratable := False; Dec(FPart1); end; end; for i := 0 to ABrick.SupportBricks.Count - 1 do ABrick.SupportBricks[i].AddTopBrick(ABrick); // Adjusts height and write brick to ground. ABrick.SetZ1(max + 1); for i := ABrick.X1 to ABrick.X2 do for j := ABrick.Y1 to ABrick.Y2 do begin FGround[i, j].Height := ABrick.Z2; FGround[i, j].TopBrick := ABrick; end; end; constructor TSandSlabs.Create; begin FBricks := TBricks.Create(TBrickComparer.Create); end; destructor TSandSlabs.Destroy; begin FBricks.Free; inherited Destroy; end; procedure TSandSlabs.ProcessDataLine(const ALine: string); begin FBricks.Add(TBrick.Create(ALine)); end; procedure TSandSlabs.Finish; var brick: TBrick; begin FBricks.Sort; InitGround; for brick in FBricks do StackBrick(brick); for brick in FBricks do Inc(FPart2, brick.CalcChainCount); end; function TSandSlabs.GetDataFileName: string; begin Result := 'sand_slabs.txt'; end; function TSandSlabs.GetPuzzleName: string; begin Result := 'Day 22: Sand Slabs'; end; end.