SessionAccordion, fixed timestamp formatting

This commit is contained in:
Rsl1122 2018-06-17 11:47:13 +03:00
parent 5184f0caf9
commit a092e39ee4
6 changed files with 187 additions and 22 deletions

View File

@ -6,10 +6,7 @@ import com.djrapitops.plan.data.store.objects.DateHolder;
import com.djrapitops.plan.data.time.WorldTimes; import com.djrapitops.plan.data.time.WorldTimes;
import com.djrapitops.plan.system.info.server.ServerInfo; import com.djrapitops.plan.system.info.server.ServerInfo;
import java.util.ArrayList; import java.util.*;
import java.util.List;
import java.util.Objects;
import java.util.UUID;
/** /**
* DataContainer for information about a player's play session. * DataContainer for information about a player's play session.
@ -71,6 +68,8 @@ public class Session extends DataContainer implements DateHolder {
putRawData(SessionKeys.SERVER_UUID, serverUUID); putRawData(SessionKeys.SERVER_UUID, serverUUID);
putRawData(SessionKeys.START, sessionStart); putRawData(SessionKeys.START, sessionStart);
putRawData(SessionKeys.END, sessionEnd); putRawData(SessionKeys.END, sessionEnd);
putRawData(SessionKeys.WORLD_TIMES, new WorldTimes(new HashMap<>()));
putRawData(SessionKeys.PLAYER_KILLS, new ArrayList<>());
putSupplier(SessionKeys.MOB_KILL_COUNT, () -> mobKills); putSupplier(SessionKeys.MOB_KILL_COUNT, () -> mobKills);
putSupplier(SessionKeys.DEATH_COUNT, () -> deaths); putSupplier(SessionKeys.DEATH_COUNT, () -> deaths);
putSupplier(SessionKeys.AFK_TIME, () -> afkTime); putSupplier(SessionKeys.AFK_TIME, () -> afkTime);

View File

@ -26,7 +26,7 @@ public class DataContainer extends HashMap<Key, Supplier> {
* @param <T> Type of the object * @param <T> Type of the object
*/ */
public <T> void putRawData(Key<T> key, T obj) { public <T> void putRawData(Key<T> key, T obj) {
super.put(key, () -> obj); putSupplier(key, () -> obj);
} }
public <T> void putSupplier(Key<T> key, Supplier<T> supplier) { public <T> void putSupplier(Key<T> key, Supplier<T> supplier) {

View File

@ -17,36 +17,33 @@ public class Formatters {
public static Formatter<DateHolder> year() { public static Formatter<DateHolder> year() {
return dateHolder -> { return dateHolder -> {
long date = dateHolder.getDate(); long date = dateHolder.getDate();
return date <= 0 ? FormatUtils.formatTimeStampYear(date) : "-"; return date > 0 ? FormatUtils.formatTimeStampYear(date) : "-";
}; };
} }
public static Formatter<DateHolder> day() { public static Formatter<DateHolder> day() {
return dateHolder -> { return dateHolder -> {
long date = dateHolder.getDate(); long date = dateHolder.getDate();
return date <= 0 ? FormatUtils.formatTimeStampDay(date) : "-"; return date > 0 ? FormatUtils.formatTimeStampDay(date) : "-";
}; };
} }
public static Formatter<DateHolder> second() { public static Formatter<DateHolder> second() {
return dateHolder -> { return dateHolder -> {
long date = dateHolder.getDate(); long date = dateHolder.getDate();
return date <= 0 ? FormatUtils.formatTimeStampSecond(date) : "-"; return date > 0 ? FormatUtils.formatTimeStampSecond(date) : "-";
}; };
} }
public static Formatter<DateHolder> clock() { public static Formatter<DateHolder> clock() {
return dateHolder -> { return dateHolder -> {
long date = dateHolder.getDate(); long date = dateHolder.getDate();
return date <= 0 ? FormatUtils.formatTimeStampClock(date) : "-"; return date > 0 ? FormatUtils.formatTimeStampClock(date) : "-";
}; };
} }
public static Formatter<DateHolder> iso8601NoClock() { public static Formatter<DateHolder> iso8601NoClock() {
return dateHolder -> { return dateHolder -> FormatUtils.formatTimeStampISO8601NoClock(dateHolder.getDate());
long date = dateHolder.getDate();
return date <= 0 ? FormatUtils.formatTimeStampISO8601NoClock(date) : "-";
};
} }
public static Formatter<Long> timeAmount() { public static Formatter<Long> timeAmount() {

View File

@ -35,6 +35,7 @@ import com.djrapitops.plan.utilities.html.graphs.calendar.PlayerCalendar;
import com.djrapitops.plan.utilities.html.graphs.pie.ServerPreferencePie; import com.djrapitops.plan.utilities.html.graphs.pie.ServerPreferencePie;
import com.djrapitops.plan.utilities.html.graphs.pie.WorldPie; import com.djrapitops.plan.utilities.html.graphs.pie.WorldPie;
import com.djrapitops.plan.utilities.html.structure.ServerAccordion; import com.djrapitops.plan.utilities.html.structure.ServerAccordion;
import com.djrapitops.plan.utilities.html.structure.SessionAccordion;
import com.djrapitops.plan.utilities.html.tables.ActionsTable; import com.djrapitops.plan.utilities.html.tables.ActionsTable;
import com.djrapitops.plan.utilities.html.tables.GeoInfoTable; import com.djrapitops.plan.utilities.html.tables.GeoInfoTable;
import com.djrapitops.plan.utilities.html.tables.NicknameTable; import com.djrapitops.plan.utilities.html.tables.NicknameTable;
@ -133,8 +134,8 @@ public class InspectPage extends Page {
SessionsMutator allSessionsMutator = new SessionsMutator(allSessions); SessionsMutator allSessionsMutator = new SessionsMutator(allSessions);
allSessions.sort(new SessionStartComparator()); allSessions.sort(new SessionStartComparator());
String[] sessionsAccordion = HtmlStructure.createSessionsTabContentInspectPage(sessionsByServerName, allSessions, uuid); SessionAccordion sessionAccordion = SessionAccordion.forPlayer(allSessions, () -> serverNames);
// TODO Session table if setting is enabled
ServerAccordion serverAccordion = new ServerAccordion(container, serverNames); ServerAccordion serverAccordion = new ServerAccordion(container, serverNames);
PlayerCalendar playerCalendar = new PlayerCalendar(allSessions, registered); PlayerCalendar playerCalendar = new PlayerCalendar(allSessions, registered);
@ -142,9 +143,9 @@ public class InspectPage extends Page {
addValue("calendarSeries", playerCalendar.toCalendarSeries()); addValue("calendarSeries", playerCalendar.toCalendarSeries());
addValue("firstDay", 1); addValue("firstDay", 1);
addValue("accordionSessions", sessionsAccordion[0]); addValue("accordionSessions", sessionAccordion.toHtml());
addValue("accordionServers", serverAccordion.toHtml()); addValue("accordionServers", serverAccordion.toHtml());
addValue("sessionTabGraphViewFunctions", sessionsAccordion[1] + serverAccordion.toViewScript()); addValue("sessionTabGraphViewFunctions", sessionAccordion.toViewScript() + serverAccordion.toViewScript());
long dayAgo = now - TimeAmount.DAY.ms(); long dayAgo = now - TimeAmount.DAY.ms();
long weekAgo = now - TimeAmount.WEEK.ms(); long weekAgo = now - TimeAmount.WEEK.ms();

View File

@ -61,6 +61,7 @@ public class HtmlStructure {
return builder.toString(); return builder.toString();
} }
@Deprecated
public static String[] createSessionsTabContentInspectPage(Map<String, List<Session>> sessions, List<Session> allSessions, UUID uuid) { public static String[] createSessionsTabContentInspectPage(Map<String, List<Session>> sessions, List<Session> allSessions, UUID uuid) {
if (Settings.DISPLAY_SESSIONS_AS_TABLE.isTrue()) { if (Settings.DISPLAY_SESSIONS_AS_TABLE.isTrue()) {
Map<UUID, List<Session>> sessionsByPlayer = new HashMap<>(); Map<UUID, List<Session>> sessionsByPlayer = new HashMap<>();

View File

@ -1,5 +1,23 @@
package com.djrapitops.plan.utilities.html.structure; package com.djrapitops.plan.utilities.html.structure;
import com.djrapitops.plan.api.PlanAPI;
import com.djrapitops.plan.data.container.Session;
import com.djrapitops.plan.data.store.keys.SessionKeys;
import com.djrapitops.plan.data.store.mutators.formatting.Formatter;
import com.djrapitops.plan.data.store.mutators.formatting.Formatters;
import com.djrapitops.plan.data.store.objects.DateHolder;
import com.djrapitops.plan.data.time.WorldTimes;
import com.djrapitops.plan.system.settings.Settings;
import com.djrapitops.plan.system.settings.theme.Theme;
import com.djrapitops.plan.system.settings.theme.ThemeVal;
import com.djrapitops.plan.utilities.html.HtmlStructure;
import com.djrapitops.plan.utilities.html.graphs.pie.WorldPie;
import com.djrapitops.plan.utilities.html.tables.KillsTable;
import com.djrapitops.plan.utilities.html.tables.SessionsTableCreator;
import java.util.*;
import java.util.function.Supplier;
/** /**
* Utility for creating Session accordion html and javascript from Session objects. * Utility for creating Session accordion html and javascript from Session objects.
* *
@ -9,24 +27,41 @@ package com.djrapitops.plan.utilities.html.structure;
public class SessionAccordion extends AbstractAccordion { public class SessionAccordion extends AbstractAccordion {
private final boolean forPlayer; private final boolean forPlayer;
private final List<Session> sessions;
private final Supplier<Map<UUID, String>> serverNamesSupplier;
private final Supplier<Map<UUID, String>> playerNamesSupplier;
private final StringBuilder viewScript; private final StringBuilder viewScript;
private final boolean appendWorldPercentage;
private int maxSessions;
private SessionAccordion(boolean forPlayer) { private SessionAccordion(boolean forPlayer, List<Session> sessions,
Supplier<Map<UUID, String>> serverNamesSupplier,
Supplier<Map<UUID, String>> playerNamesSupplier) {
super("session_accordion"); super("session_accordion");
this.forPlayer = forPlayer; this.forPlayer = forPlayer;
this.sessions = sessions;
this.serverNamesSupplier = serverNamesSupplier;
this.playerNamesSupplier = playerNamesSupplier;
viewScript = new StringBuilder(); viewScript = new StringBuilder();
maxSessions = Settings.MAX_SESSIONS.getNumber();
if (maxSessions <= 0) {
maxSessions = 50;
}
appendWorldPercentage = Settings.APPEND_WORLD_PERC.isTrue();
addElements(); addElements();
} }
public static SessionAccordion forServer() { public static SessionAccordion forServer(List<Session> sessions, Supplier<Map<UUID, String>> serverNamesSupplier,
return new SessionAccordion(false); Supplier<Map<UUID, String>> playerNamesSupplier) {
return new SessionAccordion(false, sessions, serverNamesSupplier, playerNamesSupplier);
} }
public static SessionAccordion forPlayer() { public static SessionAccordion forPlayer(List<Session> sessions, Supplier<Map<UUID, String>> serverNamesSupplier) {
return new SessionAccordion(true); return new SessionAccordion(true, sessions, serverNamesSupplier, HashMap::new);
} }
public String toViewScript() { public String toViewScript() {
@ -43,11 +78,143 @@ public class SessionAccordion extends AbstractAccordion {
} }
private void addElementsForServer() { private void addElementsForServer() {
Map<UUID, String> serverNames = serverNamesSupplier.get();
Map<UUID, String> playerNames = playerNamesSupplier.get();
Formatter<Long> timeFormatter = Formatters.timeAmount();
Formatter<DateHolder> timeStampFormatter = Formatters.year();
int i = 0;
for (Session session : sessions) {
if (i >= maxSessions) {
break;
}
String serverName = serverNames.getOrDefault(session.getValue(SessionKeys.SERVER_UUID).orElse(null), "Unknown");
String playerName = playerNames.getOrDefault(session.getValue(SessionKeys.UUID).orElse(null), "Unknown");
String sessionStart = timeStampFormatter.apply(session);
WorldTimes worldTimes = session.getValue(SessionKeys.WORLD_TIMES).orElse(new WorldTimes(new HashMap<>()));
WorldPie worldPie = new WorldPie(worldTimes);
boolean hasEnded = session.supports(SessionKeys.END);
String sessionEnd = hasEnded ? timeStampFormatter.apply(() -> session.getUnsafe(SessionKeys.END)) : "Online";
String length = (hasEnded ? "" : "(Online) ") + timeFormatter.apply(session.getValue(SessionKeys.LENGTH).orElse(0L));
String afk = (hasEnded ? "" : "(Inaccurate) ") + timeFormatter.apply(session.getValue(SessionKeys.AFK_TIME).orElse(0L));
int playerKillCount = session.getValue(SessionKeys.PLAYER_KILL_COUNT).orElse(0);
int mobKillCount = session.getValue(SessionKeys.MOB_KILL_COUNT).orElse(0);
int deaths = session.getValue(SessionKeys.DEATH_COUNT).orElse(0);
String info = appendWorldPercentage
? HtmlStructure.separateWithDots(sessionStart, SessionsTableCreator.getLongestWorldPlayed(session))
: sessionStart;
String title = HtmlStructure.separateWithDots(playerName, info) + "<span class=\"pull-right\">" + length + "</span>";
String htmlID = "" + session.getValue(SessionKeys.START).orElse(0L) + i;
String worldHtmlID = "worldPieSession" + htmlID;
String leftSide = new AccordionElementContentBuilder()
.addRowBold("teal", "clock-o", "Session Ended", sessionEnd)
.addRowBold("green", "clock-o", "Session Length", length)
.addRowBold("grey", "clock-o", "AFK", afk)
.addRowBold("green", "server", "Server", serverName)
.addBreak()
.addRowBold("red", "crosshairs", "Player Kills", playerKillCount)
.addRowBold("green", "crosshairs", "Mob Kills", mobKillCount)
.addRowBold("red", "frown-o", "Deaths", deaths)
.toHtml();
String rightSide = "<div id=\"" + worldHtmlID + "\" class=\"dashboard-donut-chart\"></div>" +
"<script>" +
"var " + worldHtmlID + "series = {name:'World Playtime',colorByPoint:true,data:" + worldPie.toHighChartsSeries() + "};" +
"var " + worldHtmlID + "gmseries = " + worldPie.toHighChartsDrilldown() + ";" +
"</script>";
viewScript.append("worldPie(")
.append(worldHtmlID).append(", ")
.append(worldHtmlID).append("series, ")
.append(worldHtmlID).append("gmseries")
.append(");");
String leftBottom = new KillsTable(session.getValue(SessionKeys.PLAYER_KILLS).orElse(new ArrayList<>())).parseHtml();
String link = PlanAPI.getInstance().getPlayerInspectPageLink(playerName);
String rightBottom = "<a target=\"_blank\" href=\"" + link + "\"><button href=\"" + link +
"\" type=\"button\" class=\"pull-right btn bg-blue waves-effect\">" +
"<i class=\"material-icons\">person</i><span>INSPECT PAGE</span></button></a>";
addElement(new AccordionElement(htmlID, title)
.setColor(Theme.getValue(ThemeVal.PARSED_SESSION_ACCORDION))
.setLeftSide(leftSide + leftBottom)
.setRightSide(rightSide + rightBottom));
i++;
}
} }
private void addElementsForPlayer() { private void addElementsForPlayer() {
Map<UUID, String> serverNames = serverNamesSupplier.get();
Formatter<Long> timeFormatter = Formatters.timeAmount();
Formatter<DateHolder> timeStampFormatter = Formatters.year();
int i = 0;
for (Session session : sessions) {
if (i >= maxSessions) {
break;
}
String serverName = serverNames.getOrDefault(session.getValue(SessionKeys.SERVER_UUID).orElse(null), "Unknown");
String sessionStart = timeStampFormatter.apply(session);
WorldTimes worldTimes = session.getValue(SessionKeys.WORLD_TIMES).orElse(new WorldTimes(new HashMap<>()));
WorldPie worldPie = new WorldPie(worldTimes);
boolean hasEnded = session.supports(SessionKeys.END);
String sessionEnd = hasEnded ? timeStampFormatter.apply(() -> session.getValue(SessionKeys.END).orElse(0L)) : "Online";
String length = (hasEnded ? "" : "(Online) ") + timeFormatter.apply(session.getValue(SessionKeys.LENGTH).orElse(0L));
String afk = (hasEnded ? "" : "(Inaccurate) ") + timeFormatter.apply(session.getValue(SessionKeys.AFK_TIME).orElse(0L));
int playerKillCount = session.getValue(SessionKeys.PLAYER_KILL_COUNT).orElse(0);
int mobKillCount = session.getValue(SessionKeys.MOB_KILL_COUNT).orElse(0);
int deaths = session.getValue(SessionKeys.DEATH_COUNT).orElse(0);
String info = appendWorldPercentage
? HtmlStructure.separateWithDots(sessionStart, SessionsTableCreator.getLongestWorldPlayed(session))
: sessionStart;
String title = HtmlStructure.separateWithDots(serverName, info) + "<span class=\"pull-right\">" + length + "</span>";
String htmlID = "" + session.getValue(SessionKeys.START).orElse(0L) + i;
String worldHtmlID = "worldPieSession" + htmlID;
String leftSide = new AccordionElementContentBuilder()
.addRowBold("teal", "clock-o", "Session Ended", sessionEnd)
.addRowBold("green", "clock-o", "Session Length", length)
.addRowBold("grey", "clock-o", "AFK", afk)
.addRowBold("grey", "server", "Server", serverName)
.addBreak()
.addRowBold("red", "crosshairs", "Player Kills", playerKillCount)
.addRowBold("green", "crosshairs", "Mob Kills", mobKillCount)
.addRowBold("red", "frown-o", "Deaths", deaths)
.toHtml();
String rightSide = "<div id=\"" + worldHtmlID + "\" class=\"dashboard-donut-chart\"></div>" +
"<script>" +
"var " + worldHtmlID + "series = {name:'World Playtime',colorByPoint:true,data:" + worldPie.toHighChartsSeries() + "};" +
"var " + worldHtmlID + "gmseries = " + worldPie.toHighChartsDrilldown() + ";" +
"</script>";
viewScript.append("worldPie(")
.append(worldHtmlID).append(", ")
.append(worldHtmlID).append("series, ")
.append(worldHtmlID).append("gmseries")
.append(");");
String leftBottom = new KillsTable(session.getValue(SessionKeys.PLAYER_KILLS).orElse(new ArrayList<>())).parseHtml();
addElement(new AccordionElement(htmlID, title)
.setColor(Theme.getValue(ThemeVal.PARSED_SESSION_ACCORDION))
.setLeftSide(leftSide + leftBottom)
.setRightSide(rightSide));
i++;
}
} }
} }