Drag Drop sorting by distance attempt
This commit is contained in:
@ -38,9 +38,11 @@
|
||||
<PackageReference Include="Serilog" Version="2.12.0" />
|
||||
</ItemGroup>
|
||||
<ItemGroup>
|
||||
<ProjectReference Include="..\Shared\View-by-Distance.Shared.csproj" />
|
||||
<ProjectReference Include="..\FaceRecognitionDotNet\FaceRecognitionDotNet.csproj" />
|
||||
<ProjectReference Include="..\Metadata\Metadata.csproj" />
|
||||
<ProjectReference Include="..\Property\Property.csproj" />
|
||||
<ProjectReference Include="..\Resize\Resize.csproj" />
|
||||
<ProjectReference Include="..\Shared\View-by-Distance.Shared.csproj" />
|
||||
</ItemGroup>
|
||||
<ItemGroup>
|
||||
<None Include="..\Instance\appsettings.json">
|
||||
|
@ -3,8 +3,11 @@ using Microsoft.WindowsAPICodePack.Shell;
|
||||
using Phares.Shared;
|
||||
using Serilog;
|
||||
using System.Diagnostics;
|
||||
using System.Globalization;
|
||||
using System.Reflection;
|
||||
using System.Text.Json;
|
||||
using View_by_Distance.Drag_Drop.Models;
|
||||
using View_by_Distance.FaceRecognitionDotNet;
|
||||
using View_by_Distance.Resize.Models;
|
||||
using View_by_Distance.Shared.Models;
|
||||
using View_by_Distance.Shared.Models.Stateless.Methods;
|
||||
@ -22,10 +25,10 @@ public partial class Form : System.Windows.Forms.Form
|
||||
private readonly string _WorkingDirectory;
|
||||
private readonly Configuration _Configuration;
|
||||
private readonly IsEnvironment _IsEnvironment;
|
||||
private readonly Dictionary<int, Item> _IdToItem;
|
||||
private readonly string _ResizeFileNameExtension;
|
||||
private readonly IConfigurationRoot _ConfigurationRoot;
|
||||
private readonly Property.Models.Configuration _PropertyConfiguration;
|
||||
private readonly Dictionary<int, MappingFromItem> _IdToMappingFromItem;
|
||||
|
||||
public Form()
|
||||
{
|
||||
@ -36,7 +39,7 @@ public partial class Form : System.Windows.Forms.Form
|
||||
string workingDirectory;
|
||||
Configuration configuration;
|
||||
IsEnvironment isEnvironment;
|
||||
_IdToMappingFromItem = new();
|
||||
_IdToItem = new();
|
||||
IConfigurationRoot configurationRoot;
|
||||
LoggerConfiguration loggerConfiguration = new();
|
||||
Property.Models.Configuration propertyConfiguration;
|
||||
@ -118,12 +121,14 @@ public partial class Form : System.Windows.Forms.Form
|
||||
Container[] containers;
|
||||
Property.Models.A_Property propertyLogic = new(_AppSettings.MaxDegreeOfParallelism, _Configuration.PropertyConfiguration, _ResizeFileNameExtension, _Configuration.Reverse);
|
||||
(_, _, _, containers) = Property.Models.Stateless.Container.GetContainers(_Configuration.PropertyConfiguration, propertyLogic);
|
||||
List<MappingFromItem> collection = Program.GetMappingFromItemCollection(_Configuration, containers);
|
||||
foreach (MappingFromItem mappingFromItem in collection)
|
||||
List<Item> collection = Program.GetItemCollection(_Configuration, containers);
|
||||
foreach (Item item in collection)
|
||||
{
|
||||
if (_IdToMappingFromItem.ContainsKey(mappingFromItem.Id))
|
||||
if (item.Property?.Id is null)
|
||||
continue;
|
||||
_IdToMappingFromItem.Add(mappingFromItem.Id, mappingFromItem);
|
||||
if (_IdToItem.ContainsKey(item.Property.Id.Value))
|
||||
continue;
|
||||
_IdToItem.Add(item.Property.Id.Value, item);
|
||||
}
|
||||
if (_Logger is null)
|
||||
throw new NullReferenceException(nameof(_Logger));
|
||||
@ -135,11 +140,34 @@ public partial class Form : System.Windows.Forms.Form
|
||||
_Logger.Debug((_PropertyConfiguration is null).ToString());
|
||||
}
|
||||
|
||||
private void RenameDirectory(string path, string searchPattern)
|
||||
public static string? GetFaceEncoding(string file)
|
||||
{
|
||||
string? result;
|
||||
List<string> results = new();
|
||||
const string comment = "Comment: ";
|
||||
if (File.Exists(file))
|
||||
{
|
||||
IReadOnlyList<MetadataExtractor.Directory> directories = MetadataExtractor.ImageMetadataReader.ReadMetadata(file);
|
||||
foreach (MetadataExtractor.Directory directory in directories)
|
||||
{
|
||||
if (directory.Name != "PNG-tEXt")
|
||||
continue;
|
||||
foreach (MetadataExtractor.Tag tag in directory.Tags)
|
||||
{
|
||||
if (tag.Name != "Textual Data" || string.IsNullOrEmpty(tag.Description))
|
||||
continue;
|
||||
if (!tag.Description.StartsWith(comment))
|
||||
continue;
|
||||
results.Add(tag.Description);
|
||||
}
|
||||
}
|
||||
}
|
||||
result = results.Any() ? results[0][comment.Length..] : null;
|
||||
return result;
|
||||
}
|
||||
|
||||
private void RenameFilesInDirectory(string directory, string searchPattern)
|
||||
{
|
||||
string[] directories = Directory.GetDirectories(path, "*", SearchOption.TopDirectoryOnly);
|
||||
foreach (string directory in directories)
|
||||
RenameDirectory(directory, searchPattern);
|
||||
int? id;
|
||||
string? message;
|
||||
string checkFile;
|
||||
@ -152,10 +180,11 @@ public partial class Form : System.Windows.Forms.Form
|
||||
_ProgressBar.Visible = true;
|
||||
bool isValidImageFormatExtension;
|
||||
string? extraLargeBitmapThumbnail;
|
||||
string[] files = Directory.GetFiles(path, searchPattern, SearchOption.TopDirectoryOnly);
|
||||
string[] files = Directory.GetFiles(directory, searchPattern, SearchOption.TopDirectoryOnly);
|
||||
_ProgressBar.Maximum = files.Length;
|
||||
foreach (string file in files)
|
||||
{
|
||||
_ProgressBar.PerformStep();
|
||||
fileHolder = new(file);
|
||||
_Lines.Add(fileHolder.NameWithoutExtension);
|
||||
isValidImageFormatExtension = _Configuration.PropertyConfiguration.ValidImageFormatExtensions.Contains(fileHolder.ExtensionLowered);
|
||||
@ -166,8 +195,6 @@ public partial class Form : System.Windows.Forms.Form
|
||||
continue;
|
||||
if (fileHolder.NameWithoutExtension.Length == 1 || fileHolder.NameWithoutExtension[1..].All(l => char.IsNumber(l)))
|
||||
continue;
|
||||
if (fileHolder.NameWithoutExtension.Length == 1 || fileHolder.NameWithoutExtension[1..].All(l => char.IsNumber(l)))
|
||||
continue;
|
||||
if (!isIgnoreExtension && isValidImageFormatExtension)
|
||||
extraLargeBitmapThumbnail = null;
|
||||
else
|
||||
@ -200,11 +227,10 @@ public partial class Form : System.Windows.Forms.Form
|
||||
if (fileHolder.DirectoryName is null)
|
||||
continue;
|
||||
}
|
||||
checkFile = Path.Combine(fileHolder.DirectoryName, $"{id}{fileHolder.ExtensionLowered}");
|
||||
checkFile = Path.Combine(fileHolder.DirectoryName, $"{id.Value}{fileHolder.ExtensionLowered}");
|
||||
if (File.Exists(checkFile))
|
||||
continue;
|
||||
File.Move(fileHolder.FullName, checkFile);
|
||||
_ProgressBar.PerformStep();
|
||||
}
|
||||
_ProgressBar.Visible = false;
|
||||
}
|
||||
@ -225,55 +251,174 @@ public partial class Form : System.Windows.Forms.Form
|
||||
return result;
|
||||
}
|
||||
|
||||
private List<(string, FaceDistance)> GetFileAndFaceDistanceCollection(string[] files)
|
||||
{
|
||||
// int? id;
|
||||
string? json;
|
||||
// string? message;
|
||||
// DateTime? dateTime;
|
||||
// FileHolder fileHolder;
|
||||
// bool isIgnoreExtension;
|
||||
// DateTime? minimumDateTime;
|
||||
FaceDistance faceDistance;
|
||||
// bool isValidImageFormatExtension;
|
||||
List<(string, FaceDistance)> collection = new();
|
||||
Shared.Models.FaceEncoding? modelsFaceEncoding;
|
||||
FaceRecognitionDotNet.FaceEncoding faceRecognitionDotNetFaceEncoding;
|
||||
_ProgressBar.Maximum = files.Length;
|
||||
foreach (string file in files)
|
||||
{
|
||||
_ProgressBar.PerformStep();
|
||||
json = Metadata.Models.Stateless.IMetadata.GetFaceEncoding(file);
|
||||
if (json is null)
|
||||
break;
|
||||
modelsFaceEncoding = JsonSerializer.Deserialize<Shared.Models.FaceEncoding>(json);
|
||||
if (modelsFaceEncoding is null)
|
||||
throw new NotSupportedException();
|
||||
faceRecognitionDotNetFaceEncoding = FaceRecognition.LoadFaceEncoding(modelsFaceEncoding.RawEncoding);
|
||||
faceDistance = new(faceRecognitionDotNetFaceEncoding);
|
||||
collection.Add(new(file, faceDistance));
|
||||
// fileHolder = new(file);
|
||||
// _Lines.Add(fileHolder.NameWithoutExtension);
|
||||
// isValidImageFormatExtension = _Configuration.PropertyConfiguration.ValidImageFormatExtensions.Contains(fileHolder.ExtensionLowered);
|
||||
// isIgnoreExtension = isValidImageFormatExtension && _Configuration.IgnoreExtensions.Contains(fileHolder.ExtensionLowered);
|
||||
// if (isIgnoreExtension || !isValidImageFormatExtension)
|
||||
// continue;
|
||||
// if (fileHolder.CreationTime is null || fileHolder.Name.Contains(fileHolder.CreationTime.Value.ToString("yy")))
|
||||
// continue;
|
||||
// if (fileHolder.LastWriteTime is null || fileHolder.Name.Contains(fileHolder.LastWriteTime.Value.ToString("yy")))
|
||||
// continue;
|
||||
// if (fileHolder.NameWithoutExtension.Length == 1 || fileHolder.NameWithoutExtension[1..].All(l => char.IsNumber(l)))
|
||||
// continue;
|
||||
// if (fileHolder.DirectoryName is null)
|
||||
// continue;
|
||||
// dateTime = IProperty.GetDateTimeFromName(fileHolder);
|
||||
// if (dateTime is not null && fileHolder.Name.Contains(dateTime.Value.ToString("yy")))
|
||||
// continue;
|
||||
// (minimumDateTime, id, message) = IProperty.Get(fileHolder);
|
||||
// if (minimumDateTime is null || id is null)
|
||||
// continue;
|
||||
// if (fileHolder.Name.Contains(minimumDateTime.Value.ToString("yy")))
|
||||
// continue;
|
||||
// if (dateTime is not null && minimumDateTime.Value != dateTime.Value)
|
||||
// continue;
|
||||
}
|
||||
return collection;
|
||||
}
|
||||
|
||||
private static List<(string File, double? Sum)> GetFileAndSum(List<(string File, FaceDistance FaceDistance)> collection)
|
||||
{
|
||||
List<(string File, double? Sum)> results = new();
|
||||
List<double?> lengths = new();
|
||||
List<FaceDistance> faceDistanceLengths;
|
||||
int take = (int)(collection.Count * .3333);
|
||||
List<FaceDistance> faceDistances = collection.Select(l => l.FaceDistance).ToList();
|
||||
foreach ((string file, FaceDistance faceDistance) in collection)
|
||||
{
|
||||
lengths.Clear();
|
||||
faceDistanceLengths = FaceRecognition.FaceDistances(faceDistances, faceDistance);
|
||||
if (faceDistanceLengths.Count != faceDistances.Count)
|
||||
throw new NotSupportedException();
|
||||
lengths.AddRange(from l in faceDistanceLengths orderby l.Length is not null, l.Length select l.Length);
|
||||
results.Add(new(file, lengths.Take(take).Sum()));
|
||||
}
|
||||
return results;
|
||||
}
|
||||
|
||||
private void ChangeDate(string directory, DateTime dateTime, string searchPattern)
|
||||
{
|
||||
string[] files = Directory.GetFiles(directory, searchPattern, SearchOption.TopDirectoryOnly);
|
||||
_ProgressBar.Step = 1;
|
||||
_ProgressBar.Value = 0;
|
||||
_ProgressBar.Visible = true;
|
||||
List<(string File, double? Sum)> results;
|
||||
List<(string File, FaceDistance FaceDistance)> collection = GetFileAndFaceDistanceCollection(files);
|
||||
_ProgressBar.Maximum = files.Length;
|
||||
if (collection.Count != files.Length)
|
||||
results = new();
|
||||
else
|
||||
results = GetFileAndSum(collection);
|
||||
if (results.Count == files.Length)
|
||||
{
|
||||
foreach ((string file, double? sum) in results)
|
||||
{
|
||||
if (sum is null)
|
||||
continue;
|
||||
File.SetCreationTime(file, dateTime.AddSeconds(sum.Value));
|
||||
}
|
||||
}
|
||||
if (results.Count == files.Length)
|
||||
{ }
|
||||
}
|
||||
|
||||
private List<DateTime> GetBirthDates(string directory)
|
||||
{
|
||||
List<DateTime> results = new();
|
||||
string[] directoryNames = IPath.GetDirectoryNames(directory);
|
||||
foreach (string directoryName in directoryNames)
|
||||
{
|
||||
if (directoryName.Length != _Configuration.PersonBirthdayFormat.Length)
|
||||
continue;
|
||||
if (!DateTime.TryParseExact(directoryName, _Configuration.PersonBirthdayFormat, CultureInfo.InvariantCulture, DateTimeStyles.None, out DateTime dateTime))
|
||||
continue;
|
||||
results.Add(dateTime);
|
||||
}
|
||||
return results;
|
||||
}
|
||||
|
||||
private void RenameFilesInDirectories(List<string> directories)
|
||||
{
|
||||
string directoryName;
|
||||
ReadOnlySpan<char> span;
|
||||
for (int i = 1; i < 3; i++)
|
||||
{
|
||||
foreach (string directory in directories)
|
||||
{
|
||||
if (directory.Length < 6)
|
||||
continue;
|
||||
directoryName = Path.GetFileName(directory);
|
||||
if (directory.Contains("!---"))
|
||||
span = "0";
|
||||
else
|
||||
span = directory.AsSpan(directory.Length - 5, 3);
|
||||
if (directoryName.Length != 1 || !int.TryParse(span, out int age))
|
||||
{
|
||||
if (i == 1)
|
||||
RenameFilesInDirectory(directory, "*Rename*");
|
||||
else if (i == 2)
|
||||
RenameFilesInDirectory(directory, "*");
|
||||
else
|
||||
continue;
|
||||
}
|
||||
else
|
||||
{
|
||||
if (i == 1)
|
||||
{
|
||||
List<DateTime> birthDates = GetBirthDates(directory);
|
||||
if (birthDates.Count != 1)
|
||||
continue;
|
||||
ChangeDate(directory, birthDates[0].AddYears(age), "*");
|
||||
}
|
||||
else
|
||||
continue;
|
||||
}
|
||||
}
|
||||
}
|
||||
File.WriteAllLines($"D:/Tmp/Phares/{DateTime.Now.Ticks}.tsv", _Lines);
|
||||
_Lines.Clear();
|
||||
}
|
||||
|
||||
void Form1_DragDrop(object? sender, DragEventArgs e)
|
||||
{
|
||||
try
|
||||
{
|
||||
if (e.Data is null)
|
||||
if (e.Data is null || e.Data.GetData(DataFormats.FileDrop) is not string[] paths)
|
||||
_TextBox.Text = string.Empty;
|
||||
else
|
||||
{
|
||||
string name;
|
||||
string[] segments;
|
||||
for (int i = 1; i < 3; i++)
|
||||
{
|
||||
if (e.Data.GetData(DataFormats.FileDrop) is not string[] paths)
|
||||
continue;
|
||||
foreach (string path in paths)
|
||||
{
|
||||
name = Path.GetFileNameWithoutExtension(path);
|
||||
Text = name;
|
||||
segments = name.Split('.');
|
||||
if (Directory.Exists(path))
|
||||
{
|
||||
if (i == 1)
|
||||
RenameDirectory(path, "*Rename*");
|
||||
else if (i == 2)
|
||||
RenameDirectory(path, "*");
|
||||
else
|
||||
continue;
|
||||
}
|
||||
else
|
||||
{
|
||||
if (i != 1)
|
||||
continue;
|
||||
if (!_IdToMappingFromItem.Any())
|
||||
LoadData();
|
||||
if (int.TryParse(segments[0], out int id) && _IdToMappingFromItem.TryGetValue(id, out MappingFromItem? value))
|
||||
{
|
||||
MappingFromItem mappingFromItem = value;
|
||||
Text = mappingFromItem.ImageFileHolder.Name;
|
||||
_TextBox.Text = mappingFromItem.ImageFileHolder.FullName;
|
||||
if (mappingFromItem.ImageFileHolder.DirectoryName is not null)
|
||||
_Logger.Information(mappingFromItem.ImageFileHolder.DirectoryName);
|
||||
if (!string.IsNullOrEmpty(mappingFromItem.ImageFileHolder.DirectoryName))
|
||||
_ = Process.Start("explorer.exe", string.Concat("\"", mappingFromItem.ImageFileHolder.DirectoryName, "\""));
|
||||
}
|
||||
}
|
||||
}
|
||||
File.WriteAllLines($"D:/Tmp/Phares/{DateTime.Now.Ticks}.tsv", _Lines);
|
||||
_Lines.Clear();
|
||||
}
|
||||
List<string> directories = GetDirectoriesOrDoDragDrop(paths);
|
||||
if (directories.Any())
|
||||
RenameFilesInDirectories(directories);
|
||||
}
|
||||
}
|
||||
catch (Exception)
|
||||
@ -282,4 +427,37 @@ public partial class Form : System.Windows.Forms.Form
|
||||
}
|
||||
}
|
||||
|
||||
private List<string> GetDirectoriesOrDoDragDrop(string[] paths)
|
||||
{
|
||||
List<string> results = new();
|
||||
string name;
|
||||
string[] segments;
|
||||
foreach (string path in paths)
|
||||
{
|
||||
name = Path.GetFileNameWithoutExtension(path);
|
||||
Text = name;
|
||||
segments = name.Split('.');
|
||||
if (Directory.Exists(path))
|
||||
{
|
||||
results.Add(path);
|
||||
results.AddRange(Directory.GetDirectories(path, "*", SearchOption.AllDirectories));
|
||||
}
|
||||
else
|
||||
{
|
||||
if (!_IdToItem.Any())
|
||||
LoadData();
|
||||
if (int.TryParse(segments[0], out int id) && _IdToItem.TryGetValue(id, out Item? item))
|
||||
{
|
||||
Text = item.ImageFileHolder.Name;
|
||||
_TextBox.Text = item.ImageFileHolder.FullName;
|
||||
if (item.ImageFileHolder.DirectoryName is not null)
|
||||
_Logger.Information(item.ImageFileHolder.DirectoryName);
|
||||
if (!string.IsNullOrEmpty(item.ImageFileHolder.DirectoryName))
|
||||
_ = Process.Start("explorer.exe", string.Concat("\"", item.ImageFileHolder.DirectoryName, "\""));
|
||||
}
|
||||
}
|
||||
}
|
||||
return results;
|
||||
}
|
||||
|
||||
}
|
@ -35,6 +35,7 @@ public class Configuration
|
||||
[Display(Name = "Override For Face Images"), Required] public bool? OverrideForFaceImages { get; set; }
|
||||
[Display(Name = "Override For Face Landmark Images"), Required] public bool? OverrideForFaceLandmarkImages { get; set; }
|
||||
[Display(Name = "Override For Resize Images"), Required] public bool? OverrideForResizeImages { get; set; }
|
||||
[Display(Name = "Person Birthday Format"), Required] public string PersonBirthdayFormat { get; set; }
|
||||
[Display(Name = "Predictor Model Name"), Required] public string PredictorModelName { get; set; }
|
||||
[Display(Name = "Properties Changed For Distance"), Required] public bool? PropertiesChangedForDistance { get; set; }
|
||||
[Display(Name = "Properties Changed For Faces"), Required] public bool? PropertiesChangedForFaces { get; set; }
|
||||
@ -103,6 +104,8 @@ public class Configuration
|
||||
throw new NullReferenceException(nameof(configuration.OverrideForFaceLandmarkImages));
|
||||
if (configuration.OverrideForResizeImages is null)
|
||||
throw new NullReferenceException(nameof(configuration.OverrideForResizeImages));
|
||||
if (configuration.PersonBirthdayFormat is null)
|
||||
throw new NullReferenceException(nameof(configuration.PersonBirthdayFormat));
|
||||
if (configuration.PropertiesChangedForDistance is null)
|
||||
throw new NullReferenceException(nameof(configuration.PropertiesChangedForDistance));
|
||||
if (configuration.PropertiesChangedForFaces is null)
|
||||
@ -154,6 +157,7 @@ public class Configuration
|
||||
configuration.OverrideForFaceImages.Value,
|
||||
configuration.OverrideForFaceLandmarkImages.Value,
|
||||
configuration.OverrideForResizeImages.Value,
|
||||
configuration.PersonBirthdayFormat,
|
||||
configuration.PredictorModelName,
|
||||
configuration.PropertiesChangedForDistance.Value,
|
||||
configuration.PropertiesChangedForFaces.Value,
|
||||
|
@ -34,6 +34,7 @@ public class Configuration
|
||||
public bool OverrideForFaceImages { init; get; }
|
||||
public bool OverrideForFaceLandmarkImages { init; get; }
|
||||
public bool OverrideForResizeImages { init; get; }
|
||||
public string PersonBirthdayFormat { init; get; }
|
||||
public string PredictorModelName { init; get; }
|
||||
public bool PropertiesChangedForDistance { init; get; }
|
||||
public bool PropertiesChangedForFaces { init; get; }
|
||||
@ -77,6 +78,7 @@ public class Configuration
|
||||
bool overrideForFaceImages,
|
||||
bool overrideForFaceLandmarkImages,
|
||||
bool overrideForResizeImages,
|
||||
string personBirthdayFormat,
|
||||
string predictorModelName,
|
||||
bool propertiesChangedForDistance,
|
||||
bool propertiesChangedForFaces,
|
||||
@ -119,6 +121,7 @@ public class Configuration
|
||||
OverrideForFaceImages = overrideForFaceImages;
|
||||
OverrideForFaceLandmarkImages = overrideForFaceLandmarkImages;
|
||||
OverrideForResizeImages = overrideForResizeImages;
|
||||
PersonBirthdayFormat = personBirthdayFormat;
|
||||
PredictorModelName = predictorModelName;
|
||||
PropertiesChangedForDistance = propertiesChangedForDistance;
|
||||
PropertiesChangedForFaces = propertiesChangedForFaces;
|
||||
|
@ -1,5 +1,4 @@
|
||||
using View_by_Distance.Shared.Models;
|
||||
using View_by_Distance.Shared.Models.Stateless.Methods;
|
||||
|
||||
namespace View_by_Distance.Drag_Drop;
|
||||
|
||||
@ -47,11 +46,10 @@ static class Program
|
||||
return result;
|
||||
}
|
||||
|
||||
public static List<MappingFromItem> GetMappingFromItemCollection(Models.Configuration configuration, Container[] containers)
|
||||
public static List<Item> GetItemCollection(Models.Configuration configuration, Container[] containers)
|
||||
{
|
||||
List<MappingFromItem> results = new();
|
||||
List<Item> results = new();
|
||||
Item[] filteredItems;
|
||||
MappingFromItem mappingFromItem;
|
||||
foreach (Container container in containers)
|
||||
{
|
||||
if (!container.Items.Any())
|
||||
@ -62,12 +60,7 @@ static class Program
|
||||
if (!filteredItems.Any())
|
||||
continue;
|
||||
foreach (Item item in filteredItems)
|
||||
{
|
||||
if (item.Property?.Id is null)
|
||||
continue;
|
||||
mappingFromItem = IMappingFromItem.GetMappingFromItem(item);
|
||||
results.Add(mappingFromItem);
|
||||
}
|
||||
results.Add(item);
|
||||
}
|
||||
return results;
|
||||
}
|
||||
|
Reference in New Issue
Block a user