mirror of
https://github.com/BentoBoxWorld/BentoBox.git
synced 2025-02-21 06:41:56 +01:00
Merge pull request #2327 from BentoBoxWorld/2320_hide_flags_in_other_world
Fix #2320. Enables hiding of flags when in another world
This commit is contained in:
commit
3c6e3d1286
@ -4,6 +4,7 @@ import java.util.Objects;
|
|||||||
|
|
||||||
import org.bukkit.Bukkit;
|
import org.bukkit.Bukkit;
|
||||||
import org.bukkit.Sound;
|
import org.bukkit.Sound;
|
||||||
|
import org.bukkit.World;
|
||||||
import org.bukkit.event.inventory.ClickType;
|
import org.bukkit.event.inventory.ClickType;
|
||||||
|
|
||||||
import world.bentobox.bentobox.BentoBox;
|
import world.bentobox.bentobox.BentoBox;
|
||||||
@ -59,6 +60,11 @@ public class CycleClick implements PanelItem.ClickHandler {
|
|||||||
|
|
||||||
@Override
|
@Override
|
||||||
public boolean onClick(Panel panel, User user2, ClickType click, int slot) {
|
public boolean onClick(Panel panel, User user2, ClickType click, int slot) {
|
||||||
|
if (panel.getWorld().isEmpty()) {
|
||||||
|
plugin.logError("Panel " + panel.getName()
|
||||||
|
+ " has no world associated with it. Please report this bug to the author.");
|
||||||
|
return true;
|
||||||
|
}
|
||||||
// This click listener is used with TabbedPanel and SettingsTabs only
|
// This click listener is used with TabbedPanel and SettingsTabs only
|
||||||
TabbedPanel tp = (TabbedPanel)panel;
|
TabbedPanel tp = (TabbedPanel)panel;
|
||||||
SettingsTab st = (SettingsTab)tp.getActiveTab();
|
SettingsTab st = (SettingsTab)tp.getActiveTab();
|
||||||
@ -67,7 +73,7 @@ public class CycleClick implements PanelItem.ClickHandler {
|
|||||||
this.user = user2;
|
this.user = user2;
|
||||||
changeOccurred = false;
|
changeOccurred = false;
|
||||||
// Permission prefix
|
// Permission prefix
|
||||||
String prefix = plugin.getIWM().getPermissionPrefix(Util.getWorld(user.getWorld()));
|
String prefix = plugin.getIWM().getPermissionPrefix(Util.getWorld(panel.getWorld().get()));
|
||||||
String reqPerm = prefix + "settings." + id;
|
String reqPerm = prefix + "settings." + id;
|
||||||
String allPerms = prefix + "settings.*";
|
String allPerms = prefix + "settings.*";
|
||||||
if (!user.hasPermission(reqPerm) && !user.hasPermission(allPerms)
|
if (!user.hasPermission(reqPerm) && !user.hasPermission(allPerms)
|
||||||
@ -91,7 +97,7 @@ public class CycleClick implements PanelItem.ClickHandler {
|
|||||||
rightClick(flag, currentRank);
|
rightClick(flag, currentRank);
|
||||||
|
|
||||||
} else if (click.equals(ClickType.SHIFT_LEFT) && user2.isOp()) {
|
} else if (click.equals(ClickType.SHIFT_LEFT) && user2.isOp()) {
|
||||||
leftShiftClick(flag);
|
leftShiftClick(flag, panel.getWorld().get());
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
} else {
|
} else {
|
||||||
@ -149,16 +155,16 @@ public class CycleClick implements PanelItem.ClickHandler {
|
|||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private void leftShiftClick(Flag flag) {
|
private void leftShiftClick(Flag flag, World world) {
|
||||||
if (!plugin.getIWM().getHiddenFlags(user.getWorld()).contains(flag.getID())) {
|
if (!plugin.getIWM().getHiddenFlags(world).contains(flag.getID())) {
|
||||||
plugin.getIWM().getHiddenFlags(user.getWorld()).add(flag.getID());
|
plugin.getIWM().getHiddenFlags(world).add(flag.getID());
|
||||||
user.getPlayer().playSound(user.getLocation(), Sound.BLOCK_GLASS_BREAK, 1F, 1F);
|
user.getPlayer().playSound(user.getLocation(), Sound.BLOCK_GLASS_BREAK, 1F, 1F);
|
||||||
} else {
|
} else {
|
||||||
plugin.getIWM().getHiddenFlags(user.getWorld()).remove(flag.getID());
|
plugin.getIWM().getHiddenFlags(world).remove(flag.getID());
|
||||||
user.getPlayer().playSound(user.getLocation(), Sound.BLOCK_NOTE_BLOCK_CHIME, 1F, 1F);
|
user.getPlayer().playSound(user.getLocation(), Sound.BLOCK_NOTE_BLOCK_CHIME, 1F, 1F);
|
||||||
}
|
}
|
||||||
// Save changes
|
// Save changes
|
||||||
plugin.getIWM().getAddon(user.getWorld()).ifPresent(GameModeAddon::saveWorldSettings);
|
plugin.getIWM().getAddon(world).ifPresent(GameModeAddon::saveWorldSettings);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1,28 +1,35 @@
|
|||||||
package world.bentobox.bentobox.api.flags.clicklisteners;
|
package world.bentobox.bentobox.api.flags.clicklisteners;
|
||||||
|
|
||||||
|
import static org.junit.Assert.assertEquals;
|
||||||
|
import static org.junit.Assert.assertFalse;
|
||||||
import static org.junit.Assert.assertNotNull;
|
import static org.junit.Assert.assertNotNull;
|
||||||
import static org.junit.Assert.assertTrue;
|
import static org.junit.Assert.assertTrue;
|
||||||
import static org.mockito.ArgumentMatchers.any;
|
import static org.mockito.ArgumentMatchers.any;
|
||||||
import static org.mockito.ArgumentMatchers.anyString;
|
import static org.mockito.ArgumentMatchers.anyString;
|
||||||
import static org.mockito.ArgumentMatchers.eq;
|
import static org.mockito.ArgumentMatchers.eq;
|
||||||
import static org.mockito.Mockito.mock;
|
import static org.mockito.Mockito.mock;
|
||||||
|
import static org.mockito.Mockito.never;
|
||||||
import static org.mockito.Mockito.times;
|
import static org.mockito.Mockito.times;
|
||||||
import static org.mockito.Mockito.verify;
|
import static org.mockito.Mockito.verify;
|
||||||
import static org.mockito.Mockito.when;
|
import static org.mockito.Mockito.when;
|
||||||
|
|
||||||
|
import java.util.ArrayList;
|
||||||
import java.util.Arrays;
|
import java.util.Arrays;
|
||||||
import java.util.HashSet;
|
import java.util.HashSet;
|
||||||
|
import java.util.List;
|
||||||
import java.util.Optional;
|
import java.util.Optional;
|
||||||
import java.util.UUID;
|
import java.util.UUID;
|
||||||
|
|
||||||
import org.bukkit.Bukkit;
|
import org.bukkit.Bukkit;
|
||||||
import org.bukkit.Location;
|
import org.bukkit.Location;
|
||||||
|
import org.bukkit.Sound;
|
||||||
import org.bukkit.World;
|
import org.bukkit.World;
|
||||||
import org.bukkit.entity.Player;
|
import org.bukkit.entity.Player;
|
||||||
import org.bukkit.event.inventory.ClickType;
|
import org.bukkit.event.inventory.ClickType;
|
||||||
import org.bukkit.inventory.Inventory;
|
import org.bukkit.inventory.Inventory;
|
||||||
import org.bukkit.plugin.PluginManager;
|
import org.bukkit.plugin.PluginManager;
|
||||||
import org.bukkit.scheduler.BukkitScheduler;
|
import org.bukkit.scheduler.BukkitScheduler;
|
||||||
|
import org.eclipse.jdt.annotation.NonNull;
|
||||||
import org.junit.After;
|
import org.junit.After;
|
||||||
import org.junit.Before;
|
import org.junit.Before;
|
||||||
import org.junit.Test;
|
import org.junit.Test;
|
||||||
@ -59,6 +66,8 @@ public class CycleClickTest {
|
|||||||
private static final Integer X = 600;
|
private static final Integer X = 600;
|
||||||
private static final Integer Y = 120;
|
private static final Integer Y = 120;
|
||||||
private static final Integer Z = 10000;
|
private static final Integer Z = 10000;
|
||||||
|
private static final int SLOT = 5;
|
||||||
|
private static final String LOCK = "LOCK";
|
||||||
@Mock
|
@Mock
|
||||||
private BentoBox plugin;
|
private BentoBox plugin;
|
||||||
private UUID uuid;
|
private UUID uuid;
|
||||||
@ -82,6 +91,9 @@ public class CycleClickTest {
|
|||||||
private SettingsTab settingsTab;
|
private SettingsTab settingsTab;
|
||||||
@Mock
|
@Mock
|
||||||
private RanksManager rm;
|
private RanksManager rm;
|
||||||
|
private List<String> hiddenFlags;
|
||||||
|
@Mock
|
||||||
|
private @NonNull Player p;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @throws java.lang.Exception - exception
|
* @throws java.lang.Exception - exception
|
||||||
@ -99,9 +111,7 @@ public class CycleClickTest {
|
|||||||
Settings s = mock(Settings.class);
|
Settings s = mock(Settings.class);
|
||||||
when(plugin.getSettings()).thenReturn(s);
|
when(plugin.getSettings()).thenReturn(s);
|
||||||
|
|
||||||
// Player
|
// User
|
||||||
Player p = mock(Player.class);
|
|
||||||
// Sometimes use Mockito.withSettings().verboseLogging()
|
|
||||||
User.setPlugin(plugin);
|
User.setPlugin(plugin);
|
||||||
when(user.isOp()).thenReturn(false);
|
when(user.isOp()).thenReturn(false);
|
||||||
uuid = UUID.randomUUID();
|
uuid = UUID.randomUUID();
|
||||||
@ -178,6 +188,7 @@ public class CycleClickTest {
|
|||||||
when(im.getIslandAt(any())).thenReturn(opIsland);
|
when(im.getIslandAt(any())).thenReturn(opIsland);
|
||||||
|
|
||||||
FlagsManager fm = mock(FlagsManager.class);
|
FlagsManager fm = mock(FlagsManager.class);
|
||||||
|
when(flag.getID()).thenReturn(LOCK);
|
||||||
when(fm.getFlag(anyString())).thenReturn(Optional.of(flag));
|
when(fm.getFlag(anyString())).thenReturn(Optional.of(flag));
|
||||||
when(plugin.getFlagsManager()).thenReturn(fm);
|
when(plugin.getFlagsManager()).thenReturn(fm);
|
||||||
|
|
||||||
@ -210,8 +221,13 @@ public class CycleClickTest {
|
|||||||
|
|
||||||
// Active tab
|
// Active tab
|
||||||
when(panel.getActiveTab()).thenReturn(settingsTab);
|
when(panel.getActiveTab()).thenReturn(settingsTab);
|
||||||
|
when(panel.getWorld()).thenReturn(Optional.of(world));
|
||||||
|
when(panel.getName()).thenReturn("name");
|
||||||
when(settingsTab.getIsland()).thenReturn(island);
|
when(settingsTab.getIsland()).thenReturn(island);
|
||||||
|
|
||||||
|
// Hidden flags
|
||||||
|
hiddenFlags = new ArrayList<>();
|
||||||
|
when(iwm.getHiddenFlags(world)).thenReturn(hiddenFlags);
|
||||||
}
|
}
|
||||||
|
|
||||||
@After
|
@After
|
||||||
@ -222,25 +238,28 @@ public class CycleClickTest {
|
|||||||
@Test
|
@Test
|
||||||
public void testNoPremission() {
|
public void testNoPremission() {
|
||||||
when(user.hasPermission(anyString())).thenReturn(false);
|
when(user.hasPermission(anyString())).thenReturn(false);
|
||||||
CycleClick udc = new CycleClick("LOCK");
|
CycleClick udc = new CycleClick(LOCK);
|
||||||
assertTrue(udc.onClick(panel, user, ClickType.LEFT, 5));
|
assertTrue(udc.onClick(panel, user, ClickType.LEFT, 5));
|
||||||
verify(user).sendMessage(eq("general.errors.no-permission"), eq("[permission]"), eq("bskyblock.settings.LOCK"));
|
verify(user).sendMessage(eq("general.errors.no-permission"), eq("[permission]"), eq("bskyblock.settings.LOCK"));
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void testUpDownClick() {
|
public void testUpDownClick() {
|
||||||
CycleClick udc = new CycleClick("LOCK");
|
CycleClick udc = new CycleClick(LOCK);
|
||||||
assertNotNull(udc);
|
assertNotNull(udc);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Test for {@link CycleClick#onClick(world.bentobox.bentobox.api.panels.Panel, User, ClickType, int)}
|
||||||
|
*/
|
||||||
@Test
|
@Test
|
||||||
public void testOnLeftClick() {
|
public void testOnLeftClick() {
|
||||||
final int SLOT = 5;
|
final int SLOT = 5;
|
||||||
CycleClick udc = new CycleClick("LOCK");
|
CycleClick udc = new CycleClick(LOCK);
|
||||||
// Rank starts at member
|
// Rank starts at member
|
||||||
// Click left
|
// Click left
|
||||||
assertTrue(udc.onClick(panel, user, ClickType.LEFT, SLOT));
|
assertTrue(udc.onClick(panel, user, ClickType.LEFT, SLOT));
|
||||||
verify(island).setFlag(eq(flag), eq(RanksManager.OWNER_RANK));
|
verify(island).setFlag(flag, RanksManager.OWNER_RANK);
|
||||||
// Check rollover
|
// Check rollover
|
||||||
// Clicking when Owner should go to Visitor
|
// Clicking when Owner should go to Visitor
|
||||||
when(island.getFlag(any())).thenReturn(RanksManager.OWNER_RANK);
|
when(island.getFlag(any())).thenReturn(RanksManager.OWNER_RANK);
|
||||||
@ -249,65 +268,114 @@ public class CycleClickTest {
|
|||||||
verify(pim, times(2)).callEvent(any(FlagProtectionChangeEvent.class));
|
verify(pim, times(2)).callEvent(any(FlagProtectionChangeEvent.class));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Test for {@link CycleClick#onClick(world.bentobox.bentobox.api.panels.Panel, User, ClickType, int)}
|
||||||
|
*/
|
||||||
@Test
|
@Test
|
||||||
public void testOnLeftClickSetMinMax() {
|
public void testOnLeftClickSetMinMax() {
|
||||||
// Provide a current rank value - coop
|
// Provide a current rank value - coop
|
||||||
when(island.getFlag(any())).thenReturn(RanksManager.COOP_RANK);
|
when(island.getFlag(any())).thenReturn(RanksManager.COOP_RANK);
|
||||||
final int SLOT = 5;
|
final int SLOT = 5;
|
||||||
CycleClick udc = new CycleClick("LOCK", RanksManager.COOP_RANK, RanksManager.MEMBER_RANK);
|
CycleClick udc = new CycleClick(LOCK, RanksManager.COOP_RANK, RanksManager.MEMBER_RANK);
|
||||||
// Rank starts at member
|
// Rank starts at member
|
||||||
// Click left
|
// Click left
|
||||||
assertTrue(udc.onClick(panel, user, ClickType.LEFT, SLOT));
|
assertTrue(udc.onClick(panel, user, ClickType.LEFT, SLOT));
|
||||||
verify(island).setFlag(eq(flag), eq(RanksManager.TRUSTED_RANK));
|
verify(island).setFlag(flag, RanksManager.TRUSTED_RANK);
|
||||||
// Check rollover
|
// Check rollover
|
||||||
// Clicking when Member should go to Coop
|
// Clicking when Member should go to Coop
|
||||||
when(island.getFlag(any())).thenReturn(RanksManager.MEMBER_RANK);
|
when(island.getFlag(any())).thenReturn(RanksManager.MEMBER_RANK);
|
||||||
assertTrue(udc.onClick(panel, user, ClickType.LEFT, SLOT));
|
assertTrue(udc.onClick(panel, user, ClickType.LEFT, SLOT));
|
||||||
verify(island).setFlag(eq(flag), eq(RanksManager.COOP_RANK));
|
verify(island).setFlag(flag, RanksManager.COOP_RANK);
|
||||||
verify(pim, times(2)).callEvent(any(FlagProtectionChangeEvent.class));
|
verify(pim, times(2)).callEvent(any(FlagProtectionChangeEvent.class));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Test for {@link CycleClick#onClick(world.bentobox.bentobox.api.panels.Panel, User, ClickType, int)}
|
||||||
|
*/
|
||||||
@Test
|
@Test
|
||||||
public void testOnRightClick() {
|
public void testOnRightClick() {
|
||||||
final int SLOT = 5;
|
final int SLOT = 5;
|
||||||
CycleClick udc = new CycleClick("LOCK");
|
CycleClick udc = new CycleClick(LOCK);
|
||||||
// Rank starts at member
|
// Rank starts at member
|
||||||
// Right click - down rank to Trusted
|
// Right click - down rank to Trusted
|
||||||
assertTrue(udc.onClick(panel, user, ClickType.RIGHT, SLOT));
|
assertTrue(udc.onClick(panel, user, ClickType.RIGHT, SLOT));
|
||||||
verify(island).setFlag(eq(flag), eq(RanksManager.TRUSTED_RANK));
|
verify(island).setFlag(flag, RanksManager.TRUSTED_RANK);
|
||||||
// Check rollover
|
// Check rollover
|
||||||
// Clicking when Visitor should go to Owner
|
// Clicking when Visitor should go to Owner
|
||||||
when(island.getFlag(any())).thenReturn(RanksManager.VISITOR_RANK);
|
when(island.getFlag(any())).thenReturn(RanksManager.VISITOR_RANK);
|
||||||
assertTrue(udc.onClick(panel, user, ClickType.RIGHT, SLOT));
|
assertTrue(udc.onClick(panel, user, ClickType.RIGHT, SLOT));
|
||||||
verify(island).setFlag(eq(flag), eq(RanksManager.OWNER_RANK));
|
verify(island).setFlag(flag, RanksManager.OWNER_RANK);
|
||||||
verify(pim, times(2)).callEvent(any(FlagProtectionChangeEvent.class));
|
verify(pim, times(2)).callEvent(any(FlagProtectionChangeEvent.class));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Test for {@link CycleClick#onClick(world.bentobox.bentobox.api.panels.Panel, User, ClickType, int)}
|
||||||
|
*/
|
||||||
@Test
|
@Test
|
||||||
public void testOnRightClickMinMaxSet() {
|
public void testOnRightClickMinMaxSet() {
|
||||||
// Provide a current rank value - coop
|
// Provide a current rank value - coop
|
||||||
when(island.getFlag(any())).thenReturn(RanksManager.TRUSTED_RANK);
|
when(island.getFlag(any())).thenReturn(RanksManager.TRUSTED_RANK);
|
||||||
final int SLOT = 5;
|
final int SLOT = 5;
|
||||||
CycleClick udc = new CycleClick("LOCK", RanksManager.COOP_RANK, RanksManager.MEMBER_RANK);
|
CycleClick udc = new CycleClick(LOCK, RanksManager.COOP_RANK, RanksManager.MEMBER_RANK);
|
||||||
// Rank starts at member
|
// Rank starts at member
|
||||||
// Right click
|
// Right click
|
||||||
assertTrue(udc.onClick(panel, user, ClickType.RIGHT, SLOT));
|
assertTrue(udc.onClick(panel, user, ClickType.RIGHT, SLOT));
|
||||||
verify(island).setFlag(eq(flag), eq(RanksManager.COOP_RANK));
|
verify(island).setFlag(flag, RanksManager.COOP_RANK);
|
||||||
// Check rollover
|
// Check rollover
|
||||||
// Clicking when Coop should go to Member
|
// Clicking when Coop should go to Member
|
||||||
when(island.getFlag(any())).thenReturn(RanksManager.COOP_RANK);
|
when(island.getFlag(any())).thenReturn(RanksManager.COOP_RANK);
|
||||||
assertTrue(udc.onClick(panel, user, ClickType.RIGHT, SLOT));
|
assertTrue(udc.onClick(panel, user, ClickType.RIGHT, SLOT));
|
||||||
verify(island).setFlag(eq(flag), eq(RanksManager.MEMBER_RANK));
|
verify(island).setFlag(flag, RanksManager.MEMBER_RANK);
|
||||||
verify(pim, times(2)).callEvent(any(FlagProtectionChangeEvent.class));
|
verify(pim, times(2)).callEvent(any(FlagProtectionChangeEvent.class));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Test for {@link CycleClick#onClick(world.bentobox.bentobox.api.panels.Panel, User, ClickType, int)}
|
||||||
|
*/
|
||||||
@Test
|
@Test
|
||||||
public void testAllClicks() {
|
public void testAllClicks() {
|
||||||
// Test all possible click types
|
// Test all possible click types
|
||||||
CycleClick udc = new CycleClick("LOCK");
|
CycleClick udc = new CycleClick(LOCK);
|
||||||
Arrays.asList(ClickType.values()).forEach(c -> assertTrue(udc.onClick(panel, user, c, 0)));
|
Arrays.asList(ClickType.values()).forEach(c -> assertTrue(udc.onClick(panel, user, c, 0)));
|
||||||
verify(pim, times(2)).callEvent(any(FlagProtectionChangeEvent.class));
|
verify(pim, times(2)).callEvent(any(FlagProtectionChangeEvent.class));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testNoWorld() {
|
||||||
|
CycleClick udc = new CycleClick(LOCK);
|
||||||
|
when(panel.getWorld()).thenReturn(Optional.empty());
|
||||||
|
assertTrue(udc.onClick(panel, user, ClickType.SHIFT_LEFT, SLOT));
|
||||||
|
verify(plugin).logError("Panel name has no world associated with it. Please report this bug to the author.");
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Test for {@link CycleClick#onClick(world.bentobox.bentobox.api.panels.Panel, User, ClickType, int)}
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void testOnShiftLeftClickNotOp() {
|
||||||
|
CycleClick udc = new CycleClick(LOCK);
|
||||||
|
// Click shift left
|
||||||
|
assertTrue(udc.onClick(panel, user, ClickType.SHIFT_LEFT, SLOT));
|
||||||
|
verify(user, never()).sendMessage(anyString());
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Test for {@link CycleClick#onClick(world.bentobox.bentobox.api.panels.Panel, User, ClickType, int)}
|
||||||
|
*/
|
||||||
|
@Test
|
||||||
|
public void testOnShiftLeftClickIsOp() {
|
||||||
|
when(user.isOp()).thenReturn(true);
|
||||||
|
CycleClick udc = new CycleClick(LOCK);
|
||||||
|
// Click shift left
|
||||||
|
assertTrue(hiddenFlags.isEmpty());
|
||||||
|
assertTrue(udc.onClick(panel, user, ClickType.SHIFT_LEFT, SLOT));
|
||||||
|
assertFalse(hiddenFlags.isEmpty());
|
||||||
|
assertEquals(LOCK, hiddenFlags.get(0));
|
||||||
|
// Click shift left again to remove flag
|
||||||
|
assertTrue(udc.onClick(panel, user, ClickType.SHIFT_LEFT, SLOT));
|
||||||
|
assertTrue(hiddenFlags.isEmpty());
|
||||||
|
// Verify sounds
|
||||||
|
verify(p).playSound(user.getLocation(), Sound.BLOCK_GLASS_BREAK, 1F, 1F);
|
||||||
|
verify(p).playSound(user.getLocation(), Sound.BLOCK_NOTE_BLOCK_CHIME, 1F, 1F);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user