using System.Drawing; namespace View_by_Distance.Shared.Models.Stateless.Methods; internal abstract class Location { internal static bool Check(int bottom, int left, int right, int top, int zCount, bool throwException) { bool result = true; if (left < 0) result = false; if (right < 0) result = false; if (right < left) result = false; if (top < 0) result = false; if (bottom < 0) result = false; if (bottom < top) result = false; if (zCount < 0) result = false; if (throwException && !result) throw new Exception(); return result; } internal static bool Check(int bottom, int height, int left, int right, int top, int width, int zCount, bool throwException) { bool result = true; if (bottom > height) result = false; if (left > width) result = false; if (right > width) result = false; if (top > height) result = false; if (zCount < 0) result = false; if (result) result = Check(bottom, left, right, top, zCount, throwException); if (throwException && !result) throw new Exception(); return result; } internal static string GetLeftPadded(int locationDigits, string value) { string result; if (value.Length == locationDigits) result = value; else if (value.Length > locationDigits) result = value[..locationDigits]; else result = value.PadLeft(locationDigits, '0'); return result; } internal static (decimal?, decimal?, decimal?, decimal?) GetHeightLeftTopWidth(int bottom, int height, int left, int right, int top, int width, int zCount) { (decimal?, decimal?, decimal?, decimal?) result; bool verified = Check(bottom, height, left, right, top, width, zCount, throwException: false); decimal t = top; decimal l = left; decimal w = right - l; decimal h = bottom - t; decimal xHeightPercentageFactored = h / height; decimal xLeftPercentageFactored = l / width; decimal xTopPercentageFactored = t / height; decimal xWidthPercentageFactored = w / width; if (!verified) result = new(null, null, null, null); else result = new(xHeightPercentageFactored, xLeftPercentageFactored, xTopPercentageFactored, xWidthPercentageFactored); return result; } internal static int GetWholePercentages(int bottom, int height, int left, int locationDigits, int right, int top, int width, int zCount) { int result; string check; bool verified = Check(bottom, height, left, right, top, width, zCount, throwException: false); int checksum = left > top ? 4 : 8; if (!verified) check = string.Concat(checksum, new string('0', locationDigits - 1)); else { decimal factor = 100; int factorMinusOne = (int)factor - 1; int length = (locationDigits - 1) / 4; decimal x = left / (decimal)width * factor; decimal y = top / (decimal)height * factor; decimal w = (right - left) / (decimal)width * factor; decimal h = (bottom - top) / (decimal)height * factor; string xPadded = x < factor ? ILocation.GetLeftPadded(length, (int)x) : ILocation.GetLeftPadded(length, factorMinusOne); string yPadded = y < factor ? ILocation.GetLeftPadded(length, (int)y) : ILocation.GetLeftPadded(length, factorMinusOne); string widthPadded = w < factor ? ILocation.GetLeftPadded(length, (int)w) : ILocation.GetLeftPadded(length, factorMinusOne); string heightPadded = h < factor ? ILocation.GetLeftPadded(length, (int)h) : ILocation.GetLeftPadded(length, factorMinusOne); check = string.Concat(checksum, xPadded, yPadded, widthPadded, heightPadded); } long value = long.Parse(check); if (value > int.MaxValue) throw new Exception(); result = (int)value; return result; } internal static int GetWholePercentages(int height, Models.Location location, int locationDigits, int width) { int result = GetWholePercentages(location.Bottom, height, location.Left, locationDigits, location.Right, location.Top, width, zCount: 1); return result; } internal static int GetConfidencePercent(int faceConfidencePercent, double confidence) { int result = (int)(confidence * faceConfidencePercent); return result; } internal static RectangleF? GetPercentagesRectangle(int locationDigits, string wholePercentages) { RectangleF? result; int length = (locationDigits - 1) / 4; string[] segments = [ wholePercentages[..1], wholePercentages.Substring(1, length), wholePercentages.Substring(3, length), wholePercentages.Substring(5, length), wholePercentages.Substring(7, length) ]; if (string.Join(string.Empty, segments) != wholePercentages) result = null; else { if (!int.TryParse(segments[1], out int xWholePercent) || !int.TryParse(segments[2], out int yWholePercent) || !int.TryParse(segments[3], out int wWholePercent) || !int.TryParse(segments[4], out int hWholePercent)) result = null; else { float factor = 100; result = new(xWholePercent / factor, yWholePercent / factor, wWholePercent / factor, hWholePercent / factor); } } return result; } internal static Rectangle? GetRectangle(int locationDigits, Models.OutputResolution outputResolution, string wholePercentages) { Rectangle? result; if (wholePercentages.Length != locationDigits || wholePercentages[0] is not '4' and not '8') throw new NotSupportedException("Old way has been removed!"); (int width, int height) = OutputResolution.Get(outputResolution); RectangleF? rectangle = GetPercentagesRectangle(locationDigits, wholePercentages); if (rectangle is null) result = null; else { result = new((int)(rectangle.Value.X * width), (int)(rectangle.Value.Y * height), (int)(rectangle.Value.Width * width), (int)(rectangle.Value.Height * height)); } if (result is null) throw new NullReferenceException(nameof(result)); return result; } private static bool Matches(Models.OutputResolution outputResolution, DatabaseFile databaseFile) { bool result = outputResolution.Height == databaseFile.FileHeight && outputResolution.Width == databaseFile.FileWidth; return result; } internal static RectangleF? GetPercentagesRectangle(DatabaseFile databaseFile, Marker marker, Models.OutputResolution outputResolution) { RectangleF? result; bool matches = Matches(outputResolution, databaseFile); if (!matches) result = null; else result = new(marker.X, marker.Y, marker.W, marker.H); return result; } private static Models.Location? GetLocation(DatabaseFile databaseFile, Marker marker, RectangleF rectangle) { Models.Location? result; int top = (int)Math.Ceiling(rectangle.Top * databaseFile.FileHeight); int left = (int)Math.Ceiling(rectangle.Left * databaseFile.FileWidth); int right = (int)Math.Ceiling(rectangle.Right * databaseFile.FileWidth); int bottom = (int)Math.Ceiling(rectangle.Bottom * databaseFile.FileHeight); bool verified = Check(bottom, databaseFile.FileHeight, left, right, top, databaseFile.FileWidth, zCount: 1, throwException: false); if (!verified) result = null; else result = new(bottom, marker.Score / 100, left, right, top); return result; } internal static Models.Location? GetLocation(int height, Rectangle rectangle, int width) { Models.Location? result; double confidence = 0; bool verified = Check(rectangle.Bottom, height, rectangle.Left, rectangle.Right, rectangle.Top, width, zCount: 1, throwException: false); if (!verified) result = null; else result = new(rectangle.Bottom, confidence, rectangle.Left, rectangle.Right, rectangle.Top); return result; } internal static Models.Location? GetLocation(DatabaseFile databaseFile, Marker marker, Models.OutputResolution outputResolution) { Models.Location? result; RectangleF? rectangle = GetPercentagesRectangle(databaseFile, marker, outputResolution); if (rectangle is null) result = null; else result = GetLocation(databaseFile, marker, rectangle.Value); return result; } internal static float? GetIntersectPercent(RectangleF rectangleA, float? areaA, RectangleF rectangleB) { float? result; if (rectangleA.Equals(rectangleB)) result = 1; else { float intersectArea; RectangleF intersectRectangle; areaA ??= rectangleA.Width * rectangleA.Height; float areaB = rectangleB.Width * rectangleB.Height; bool check = areaA > areaB; if (check) intersectRectangle = RectangleF.Intersect(rectangleB, rectangleA); else intersectRectangle = RectangleF.Intersect(rectangleA, rectangleB); intersectArea = intersectRectangle.Width * intersectRectangle.Height; if (check) result = intersectArea / areaA; else result = intersectArea / areaB; } return result; } internal static List GetLocations(List faces, List mappingFromPhotoPrismCollection, float rectangleIntersectMinimum) { List results = []; bool any; bool matches; float? percent; float prismArea; int width, height; Models.Location? location; RectangleF? prismRectangle; int dlibLocationWholePercentages; RectangleF? dlibPercentagesRectangle; Models.OutputResolution? outputResolution = null; foreach (Models.Face face in faces) { if (face.Location is null || face.OutputResolution is null) continue; results.Add(face.Location); outputResolution ??= face.OutputResolution; } int before = results.Count; foreach (MappingFromPhotoPrism mappingFromPhotoPrism in mappingFromPhotoPrismCollection) { if (outputResolution is null) break; matches = Matches(outputResolution, mappingFromPhotoPrism.DatabaseFile); if (!matches) break; foreach (Marker marker in mappingFromPhotoPrism.Markers) { any = false; prismRectangle = GetPercentagesRectangle(mappingFromPhotoPrism.DatabaseFile, marker, outputResolution); if (prismRectangle is null) break; prismArea = prismRectangle.Value.Width * prismRectangle.Value.Height; location = GetLocation(mappingFromPhotoPrism.DatabaseFile, marker, prismRectangle.Value); if (location is null) break; foreach (Models.Face face in faces) { if (any) continue; if (face.Location is null || face.OutputResolution is null) continue; (width, height) = OutputResolution.Get(face.OutputResolution); dlibLocationWholePercentages = GetWholePercentages(height, face.Location, Stateless.ILocation.Digits, width); dlibPercentagesRectangle = GetPercentagesRectangle(Stateless.ILocation.Digits, dlibLocationWholePercentages.ToString()); if (dlibPercentagesRectangle is null) continue; percent = GetIntersectPercent(prismRectangle.Value, prismArea, dlibPercentagesRectangle.Value); if (percent is null || percent < rectangleIntersectMinimum) continue; if (!any) any = true; break; } if (!any) results.Add(location); } } if (before == results.Count) results.Clear(); return results; } internal static List FilterByIntersect(Models.Face[] faces, float rectangleIntersectMinimum, int wholePercentages) { List results = []; float? percent; int width, height; int faceLocationWholePercentages; RectangleF? facePercentagesRectangle; RectangleF? sourceRectangle = GetPercentagesRectangle(Stateless.ILocation.Digits, wholePercentages.ToString()); float? sourceArea = sourceRectangle is null ? null : sourceRectangle.Value.Width * sourceRectangle.Value.Height; foreach (Models.Face face in faces) { if (sourceRectangle is null || sourceArea is null) continue; if (face.Location is null || face.OutputResolution is null) continue; (width, height) = OutputResolution.Get(face.OutputResolution); faceLocationWholePercentages = GetWholePercentages(height, face.Location, Stateless.ILocation.Digits, width); facePercentagesRectangle = GetPercentagesRectangle(Stateless.ILocation.Digits, faceLocationWholePercentages.ToString()); if (facePercentagesRectangle is null) continue; percent = GetIntersectPercent(sourceRectangle.Value, sourceArea.Value, facePercentagesRectangle.Value); if (percent is null || percent < rectangleIntersectMinimum) continue; results.Add(face); } return results; } }