diff --git a/ui/core/proto_utils/logs_parser.tsx b/ui/core/proto_utils/logs_parser.tsx index 351c7305e4..523c91cada 100644 --- a/ui/core/proto_utils/logs_parser.tsx +++ b/ui/core/proto_utils/logs_parser.tsx @@ -1,7 +1,7 @@ // eslint-disable-next-line @typescript-eslint/no-unused-vars import { element } from 'tsx-vanilla'; -import { RaidSimResult , ResourceType } from '../proto/api.js'; +import { RaidSimResult, ResourceType } from '../proto/api.js'; import { bucket, getEnumValues, stringComparator, sum } from '../utils.js'; import { ActionId } from './action_id.js'; import { resourceNames, stringToResourceType } from './names.js'; @@ -26,7 +26,12 @@ export class Entity { } equals(other: Entity) { - return this.isTarget == other.isTarget && this.isPet == other.isPet && this.index == other.index && this.name == other.name; + return ( + this.isTarget == other.isTarget && + this.isPet == other.isPet && + this.index == other.index && + this.name == other.name + ); } toString(): string { @@ -43,7 +48,9 @@ export class Entity { if (this.isTarget) { return `[Target ${this.index + 1}]`; } else if (this.isPet) { - return `[${this.ownerName} ${this.index + 1}] - ${this.name}`; + return `[${this.ownerName} ${this.index + 1}] - ${ + this.name + }`; } else { return `[${this.name} ${this.index + 1}]`; } @@ -54,7 +61,8 @@ export class Entity { // 'Target 1' if a target, // 'PlayerName (#1)' if a player, or // 'PlayerName (#1) - PetName' if a pet. - static parseRegex = /\[(Target (\d+))|(([a-zA-Z0-9]+) \(#(\d+)\) - ([a-zA-Z0-9\s]+))|(([a-zA-Z0-9\s]+) \(#(\d+)\))\]/g; + static parseRegex = + /\[(Target (\d+))|(([a-zA-Z0-9]+) \(#(\d+)\) - ([a-zA-Z0-9\s]+))|(([a-zA-Z0-9\s]+) \(#(\d+)\))\]/g; static parseAll(str: string): Array { return Array.from(str.matchAll(Entity.parseRegex)).map(match => { if (match[1]) { @@ -71,13 +79,13 @@ export class Entity { } interface SimLogParams { - raw: string, - logIndex: number, - timestamp: number, - source: Entity | null, - target: Entity | null, - actionId: ActionId | null, - threat: number, + raw: string; + logIndex: number; + timestamp: number; + source: Entity | null; + target: Entity | null; + actionId: ActionId | null; + threat: number; } export class SimLog { @@ -155,16 +163,25 @@ export class SimLog { const seconds = Math.floor(positiveTimestamp - minutes * 60); const milliseconds = ((positiveTimestamp - Math.floor(positiveTimestamp)) * 1000).toFixed(); - let formatted = `${String(minutes).padStart(2, '0')}:${String(seconds).padStart(2, '0')}:${String(milliseconds).padStart(3, '0')}`; + let formatted = `${String(minutes).padStart(2, '0')}:${String(seconds).padStart( + 2, + '0', + )}:${String(milliseconds).padStart(3, '0')}`; if (this.timestamp < 0) { - formatted = `-${formatted}` + formatted = `-${formatted}`; } - return formatted + return formatted; } protected newActionIdLink(): string { const iconElem = ; - const actionAnchor = {iconElem} {this.actionId!.name}; + const actionAnchor = ( + + + {iconElem} {this.actionId!.name} + + + ); this.actionId?.setBackground(iconElem as HTMLAnchorElement); this.actionId?.setWowheadHref(actionAnchor as HTMLAnchorElement); return actionAnchor.outerHTML; @@ -173,46 +190,50 @@ export class SimLog { static async parseAll(result: RaidSimResult): Promise> { const lines = result.logs.split('\n'); - return Promise.all(lines.map((line, lineIndex) => { - const params: SimLogParams = { - raw: line, - logIndex: lineIndex, - timestamp: 0, - source: null, - target: null, - actionId: null, - threat: 0, - }; - - const threatMatch = line.match(/ \(Threat: (-?[0-9]+\.[0-9]+)\)/); - if (threatMatch) { - params.threat = parseFloat(threatMatch[1]); - line = line.substring(0, threatMatch.index); - } - - const match = line.match(/\[(-?[0-9]+\.[0-9]+)\]\w*(.*)/); - if (!match || !match[1]) { - return new SimLog(params); - } + return Promise.all( + lines.map((line, lineIndex) => { + const params: SimLogParams = { + raw: line, + logIndex: lineIndex, + timestamp: 0, + source: null, + target: null, + actionId: null, + threat: 0, + }; - params.timestamp = parseFloat(match[1]); - const remainder = match[2]; + const threatMatch = line.match(/ \(Threat: (-?[0-9]+\.[0-9]+)\)/); + if (threatMatch) { + params.threat = parseFloat(threatMatch[1]); + line = line.substring(0, threatMatch.index); + } - const entities = Entity.parseAll(remainder); - params.source = entities[0] || null; - params.target = entities[1] || null; + const match = line.match(/\[(-?[0-9]+\.[0-9]+)\]\w*(.*)/); + if (!match || !match[1]) { + return new SimLog(params); + } - // Order from most to least common to reduce number of checks. - return DamageDealtLog.parse(params) - || ResourceChangedLog.parse(params) - || AuraEventLog.parse(params) - || AuraStacksChangeLog.parse(params) - || MajorCooldownUsedLog.parse(params) - || CastBeganLog.parse(params) - || CastCompletedLog.parse(params) - || StatChangeLog.parse(params) - || Promise.resolve(new SimLog(params)); - })); + params.timestamp = parseFloat(match[1]); + const remainder = match[2]; + + const entities = Entity.parseAll(remainder); + params.source = entities[0] || null; + params.target = entities[1] || null; + + // Order from most to least common to reduce number of checks. + return ( + DamageDealtLog.parse(params) || + ResourceChangedLog.parse(params) || + AuraEventLog.parse(params) || + AuraStacksChangeLog.parse(params) || + MajorCooldownUsedLog.parse(params) || + CastBeganLog.parse(params) || + CastCompletedLog.parse(params) || + StatChangeLog.parse(params) || + Promise.resolve(new SimLog(params)) + ); + }), + ); } isDamageDealt(): this is DamageDealtLog { @@ -248,7 +269,9 @@ export class SimLog { } // Group events that happen at the same time. - static groupDuplicateTimestamps(logs: Array): Array> { + static groupDuplicateTimestamps( + logs: Array, + ): Array> { const grouped: Array> = []; let curGroup: Array = []; @@ -284,7 +307,22 @@ export class DamageDealtLog extends SimLog { readonly partialResist2_4: boolean; readonly partialResist3_4: boolean; - constructor(params: SimLogParams, amount: number, type: string, miss: boolean, crit: boolean, crush: boolean, glance: boolean, dodge: boolean, parry: boolean, block: boolean, tick: boolean, partialResist1_4: boolean, partialResist2_4: boolean, partialResist3_4: boolean) { + constructor( + params: SimLogParams, + amount: number, + type: string, + miss: boolean, + crit: boolean, + crush: boolean, + glance: boolean, + dodge: boolean, + parry: boolean, + block: boolean, + tick: boolean, + partialResist1_4: boolean, + partialResist2_4: boolean, + partialResist3_4: boolean, + ) { super(params); this.amount = amount; this.type = type; @@ -323,15 +361,25 @@ export class DamageDealtLog extends SimLog { result += 'Shield '; } - result += this.miss ? 'Miss' - : this.dodge ? 'Dodge' - : this.parry ? 'Parry' - : this.glance ? 'Glance' - : this.block ? (this.crit ? 'Critical Block' : 'Block') - : this.crit ? 'Crit' - : this.crush ? 'Crush' - : this.tick ? 'Tick' - : 'Hit'; + result += this.miss + ? 'Miss' + : this.dodge + ? 'Dodge' + : this.parry + ? 'Parry' + : this.glance + ? 'Glance' + : this.block + ? this.crit + ? 'Critical Block' + : 'Block' + : this.crit + ? 'Crit' + : this.crush + ? 'Crush' + : this.tick + ? 'Tick' + : 'Hit'; result += ' ' + this.target?.toHTMLString(); if (!this.miss && !this.dodge && !this.parry) { @@ -343,41 +391,48 @@ export class DamageDealtLog extends SimLog { } else if (this.partialResist3_4) { result += ' (75% Resist)'; } - result += '.' + result += '.'; } return result; } toString(includeTimestamp = true): string { const threatPostfix = this.source?.isTarget ? '' : ` (${this.threat.toFixed(2)} Threat)`; - return `${this.toStringPrefix(includeTimestamp)} ${this.newActionIdLink()} ${this.resultString()}${threatPostfix}`; + return `${this.toStringPrefix( + includeTimestamp, + )} ${this.newActionIdLink()} ${this.resultString()}${threatPostfix}`; } static parse(params: SimLogParams): Promise | null { - const match = params.raw.match(/] (.*?) (tick )?((Miss)|(Hit)|(CriticalBlock)|(Crit)|(Crush)|(Glance)|(Dodge)|(Parry)|(Block))( \((\d+)% Resist\))?( for (\d+\.\d+) ((damage)|(healing)|(shielding)))?/); + const match = params.raw.match( + /] (.*?) (tick )?((Miss)|(Hit)|(CriticalBlock)|(Crit)|(Crush)|(Glance)|(Dodge)|(Parry)|(Block))( \((\d+)% Resist\))?( for (\d+\.\d+) ((damage)|(healing)|(shielding)))?/, + ); if (match) { - return ActionId.fromLogString(match[1]).fill(params.source?.index).then(cause => { - params.actionId = cause; - - const amount = match[16] ? parseFloat(match[16]) : 0; - const type = match[17] || ''; - - return new DamageDealtLog( - params, - amount, - type, - match[3] == 'Miss', - match[3] == 'Crit' || match[3] == 'CriticalBlock', - match[3] == 'Crush', - match[3] == 'Glance', - match[3] == 'Dodge', - match[3] == 'Parry', - match[3] == 'Block' || match[3] == 'CriticalBlock', - Boolean(match[2]) && match[2].includes('tick'), - match[14] == '25', - match[14] == '50', - match[14] == '75'); - }); + return ActionId.fromLogString(match[1]) + .fill(params.source?.index) + .then(cause => { + params.actionId = cause; + + const amount = match[16] ? parseFloat(match[16]) : 0; + const type = match[17] || ''; + + return new DamageDealtLog( + params, + amount, + type, + match[3] == 'Miss', + match[3] == 'Crit' || match[3] == 'CriticalBlock', + match[3] == 'Crush', + match[3] == 'Glance', + match[3] == 'Dodge', + match[3] == 'Parry', + match[3] == 'Block' || match[3] == 'CriticalBlock', + Boolean(match[2]) && match[2].includes('tick'), + match[14] == '25', + match[14] == '50', + match[14] == '75', + ); + }); } else { return null; } @@ -427,15 +482,19 @@ export class DpsLog extends SimLog { console.warn('NaN dps!'); } - return new DpsLog({ - raw: '', - logIndex: ddLogGroup[0].logIndex, - timestamp: ddLogGroup[0].timestamp, - source: ddLogGroup[0].source, - target: null, - actionId: null, - threat: 0, - }, dps, ddLogGroup); + return new DpsLog( + { + raw: '', + logIndex: ddLogGroup[0].logIndex, + timestamp: ddLogGroup[0].timestamp, + source: ddLogGroup[0].source, + target: null, + actionId: null, + threat: 0, + }, + dps, + ddLogGroup, + ); }); } } @@ -445,7 +504,12 @@ export class ThreatLogGroup extends SimLog { readonly threatAfter: number; readonly logs: Array; - constructor(params: SimLogParams, threatBefore: number, threatAfter: number, logs: Array) { + constructor( + params: SimLogParams, + threatBefore: number, + threatAfter: number, + logs: Array, + ) { super(params); this.threatBefore = threatBefore; this.threatAfter = threatAfter; @@ -469,7 +533,8 @@ export class ThreatLogGroup extends SimLog { }, curThreat, curThreat + newThreat, - logGroup); + logGroup, + ); curThreat += newThreat; return threatLog; @@ -490,17 +555,26 @@ export class AuraEventLog extends SimLog { } toString(includeTimestamp = true): string { - return `${this.toStringPrefix(includeTimestamp)} Aura ${this.isGained ? 'gained' : this.isFaded ? 'faded' : 'refreshed'}: ${this.newActionIdLink()}.`; + return `${this.toStringPrefix(includeTimestamp)} Aura ${ + this.isGained ? 'gained' : this.isFaded ? 'faded' : 'refreshed' + }: ${this.newActionIdLink()}.`; } static parse(params: SimLogParams): Promise | null { const match = params.raw.match(/Aura ((gained)|(faded)|(refreshed)): (.*)/); if (match && match[5]) { - return ActionId.fromLogString(match[5]).fill(params.source?.index).then(aura => { - params.actionId = aura; - const event = match[1]; - return new AuraEventLog(params, event == 'gained', event == 'faded', event == 'refreshed'); - }); + return ActionId.fromLogString(match[5]) + .fill(params.source?.index) + .then(aura => { + params.actionId = aura; + const event = match[1]; + return new AuraEventLog( + params, + event == 'gained', + event == 'faded', + event == 'refreshed', + ); + }); } else { return null; } @@ -518,16 +592,20 @@ export class AuraStacksChangeLog extends SimLog { } toString(includeTimestamp = true): string { - return `${this.toStringPrefix(includeTimestamp)} ${this.newActionIdLink()} stacks: ${this.oldStacks} → ${this.newStacks}.`; + return `${this.toStringPrefix(includeTimestamp)} ${this.newActionIdLink()} stacks: ${ + this.oldStacks + } → ${this.newStacks}.`; } static parse(params: SimLogParams): Promise | null { const match = params.raw.match(/(.*) stacks: ([0-9]+) --> ([0-9]+)/); if (match && match[1]) { - return ActionId.fromLogString(match[1]).fill(params.source?.index).then(aura => { - params.actionId = aura; - return new AuraStacksChangeLog(params, parseInt(match[2]), parseInt(match[3])); - }); + return ActionId.fromLogString(match[1]) + .fill(params.source?.index) + .then(aura => { + params.actionId = aura; + return new AuraStacksChangeLog(params, parseInt(match[2]), parseInt(match[3])); + }); } else { return null; } @@ -546,8 +624,15 @@ export class AuraUptimeLog extends SimLog { this.stacksChange = stacksChange; } - static fromLogs(logs: Array, entity: Entity, encounterDuration: number): Array { - const unmatchedGainedLogs: Array<{ gained: AuraEventLog, stacks: Array }> = []; + static fromLogs( + logs: Array, + entity: Entity, + encounterDuration: number, + ): Array { + const unmatchedGainedLogs: Array<{ + gained: AuraEventLog; + stacks: Array; + }> = []; const uptimeLogs: Array = []; logs.forEach((log: SimLog) => { @@ -556,7 +641,9 @@ export class AuraUptimeLog extends SimLog { } if (log.isAuraStacksChange()) { - const matchingGainedIdx = unmatchedGainedLogs.findIndex(gainedLog => gainedLog.gained.actionId!.equals(log.actionId!)); + const matchingGainedIdx = unmatchedGainedLogs.findIndex(gainedLog => + gainedLog.gained.actionId!.equals(log.actionId!), + ); if (matchingGainedIdx == -1) { console.warn('Unmatched aura stacks change log: ' + log.actionId!.name); return; @@ -574,22 +661,33 @@ export class AuraUptimeLog extends SimLog { return; } - const matchingGainedIdx = unmatchedGainedLogs.findIndex(gainedLog => gainedLog.gained.actionId!.equals(log.actionId!)); + const matchingGainedIdx = unmatchedGainedLogs.findIndex(gainedLog => + gainedLog.gained.actionId!.equals(log.actionId!), + ); if (matchingGainedIdx == -1) { console.warn('Unmatched aura faded log: ' + log.actionId!.name); return; } - const { gained: gainedLog, stacks: stacksChangeLogs } = unmatchedGainedLogs.splice(matchingGainedIdx, 1)[0]; - - uptimeLogs.push(new AuraUptimeLog({ - raw: log.raw, - logIndex: gainedLog.logIndex, - timestamp: gainedLog.timestamp, - source: log.source, - target: log.target, - actionId: gainedLog.actionId, - threat: gainedLog.threat, - }, log.timestamp, stacksChangeLogs)); + const { gained: gainedLog, stacks: stacksChangeLogs } = unmatchedGainedLogs.splice( + matchingGainedIdx, + 1, + )[0]; + + uptimeLogs.push( + new AuraUptimeLog( + { + raw: log.raw, + logIndex: gainedLog.logIndex, + timestamp: gainedLog.timestamp, + source: log.source, + target: log.target, + actionId: gainedLog.actionId, + threat: gainedLog.threat, + }, + log.timestamp, + stacksChangeLogs, + ), + ); if (log.isRefreshed) { unmatchedGainedLogs.push({ gained: log, stacks: [] }); @@ -599,15 +697,21 @@ export class AuraUptimeLog extends SimLog { // Auras active at the end won't have a faded log, so need to add them separately. unmatchedGainedLogs.forEach(unmatchedLog => { const { gained: gainedLog, stacks: stacksChangeLogs } = unmatchedLog; - uptimeLogs.push(new AuraUptimeLog({ - raw: gainedLog.raw, - logIndex: gainedLog.logIndex, - timestamp: gainedLog.timestamp, - source: gainedLog.source, - target: gainedLog.target, - actionId: gainedLog.actionId, - threat: gainedLog.threat, - }, encounterDuration, stacksChangeLogs)); + uptimeLogs.push( + new AuraUptimeLog( + { + raw: gainedLog.raw, + logIndex: gainedLog.logIndex, + timestamp: gainedLog.timestamp, + source: gainedLog.source, + target: gainedLog.target, + actionId: gainedLog.actionId, + threat: gainedLog.threat, + }, + encounterDuration, + stacksChangeLogs, + ), + ); }); uptimeLogs.sort((a, b) => a.gainedAt - b.gainedAt); @@ -620,7 +724,10 @@ export class AuraUptimeLog extends SimLog { let auraLogsIndex = 0; logs.forEach(log => { - while (auraLogsIndex < auraLogs.length && auraLogs[auraLogsIndex].gainedAt <= log.timestamp) { + while ( + auraLogsIndex < auraLogs.length && + auraLogs[auraLogsIndex].gainedAt <= log.timestamp + ) { curAuras.push(auraLogs[auraLogsIndex]); auraLogsIndex++; } @@ -639,7 +746,13 @@ export class ResourceChangedLog extends SimLog { readonly valueAfter: number; readonly isSpend: boolean; - constructor(params: SimLogParams, resourceType: ResourceType, valueBefore: number, valueAfter: number, isSpend: boolean) { + constructor( + params: SimLogParams, + resourceType: ResourceType, + valueBefore: number, + valueAfter: number, + isSpend: boolean, + ) { super(params); this.resourceType = resourceType; this.valueBefore = valueBefore; @@ -651,11 +764,23 @@ export class ResourceChangedLog extends SimLog { const signedDiff = (this.valueAfter - this.valueBefore) * (this.isSpend ? -1 : 1); const isHealth = this.resourceType == ResourceType.ResourceTypeHealth; - const verb = isHealth ? (this.isSpend ? 'Lost' : 'Recovered') : (this.isSpend ? 'Spent' : 'Gained'); - const resourceName = resourceNames.get(this.resourceType)! - const resourceKlass = `resource-${resourceName.replace(/\s/g, "-").toLowerCase()}`; - - return `${this.toStringPrefix(includeTimestamp)} ${verb} ${signedDiff.toFixed(1)} ${resourceName} from ${this.newActionIdLink()}. (${this.valueBefore.toFixed(1)} → ${this.valueAfter.toFixed(1)})`; + const verb = isHealth + ? this.isSpend + ? 'Lost' + : 'Recovered' + : this.isSpend + ? 'Spent' + : 'Gained'; + const resourceName = resourceNames.get(this.resourceType)!; + const resourceKlass = `resource-${resourceName.replace(/\s/g, '-').toLowerCase()}`; + + return `${this.toStringPrefix( + includeTimestamp, + )} ${verb} ${signedDiff.toFixed( + 1, + )} ${resourceName} from ${this.newActionIdLink()}. (${this.valueBefore.toFixed( + 1, + )} → ${this.valueAfter.toFixed(1)})`; } resultString(): string { @@ -668,13 +793,23 @@ export class ResourceChangedLog extends SimLog { } static parse(params: SimLogParams): Promise | null { - const match = params.raw.match(/((Gained)|(Spent)) \d+\.?\d* ((health)|(mana)|(energy)|(focus)|(rage)|(combo points)) from (.*) \((\d+\.?\d*) --> (\d+\.?\d*)\)/); + const match = params.raw.match( + /((Gained)|(Spent)) \d+\.?\d* ((health)|(mana)|(energy)|(focus)|(rage)|(combo points)|(runic power)|(blood rune)|(frost rune)|(unholy rune)|(death rune)) from (.*) \((\d+\.?\d*) --> (\d+\.?\d*)\)/, + ); if (match) { const resourceType = stringToResourceType(match[4]); - return ActionId.fromLogString(match[11]).fill(params.source?.index).then(cause => { - params.actionId = cause; - return new ResourceChangedLog(params, resourceType, parseFloat(match[12]), parseFloat(match[13]), match[1] == 'Spent'); - }); + return ActionId.fromLogString(match[16]) + .fill(params.source?.index) + .then(cause => { + params.actionId = cause; + return new ResourceChangedLog( + params, + resourceType, + parseFloat(match[17]), + parseFloat(match[18]), + match[1] == 'Spent', + ); + }); } else { return null; } @@ -687,7 +822,13 @@ export class ResourceChangedLogGroup extends SimLog { readonly valueAfter: number; readonly logs: Array; - constructor(params: SimLogParams, resourceType: ResourceType, valueBefore: number, valueAfter: number, logs: Array) { + constructor( + params: SimLogParams, + resourceType: ResourceType, + valueBefore: number, + valueAfter: number, + logs: Array, + ) { super(params); this.resourceType = resourceType; this.valueBefore = valueBefore; @@ -696,32 +837,44 @@ export class ResourceChangedLogGroup extends SimLog { } toString(includeTimestamp = true): string { - return `${this.toStringPrefix(includeTimestamp)} ${resourceNames.get(this.resourceType)}: ${this.valueBefore.toFixed(1)} → ${this.valueAfter.toFixed(1)}`; + return `${this.toStringPrefix(includeTimestamp)} ${resourceNames.get( + this.resourceType, + )}: ${this.valueBefore.toFixed(1)} → ${this.valueAfter.toFixed(1)}`; } static fromLogs(logs: Array): Record> { - const allResourceChangedLogs = logs.filter((log): log is ResourceChangedLog => log.isResourceChanged()); + const allResourceChangedLogs = logs.filter((log): log is ResourceChangedLog => + log.isResourceChanged(), + ); const results: Partial>> = {}; - const resourceTypes = (getEnumValues(ResourceType) as Array).filter(val => val != ResourceType.ResourceTypeNone); + const resourceTypes = (getEnumValues(ResourceType) as Array).filter( + val => val != ResourceType.ResourceTypeNone, + ); resourceTypes.forEach(resourceType => { - const resourceChangedLogs = allResourceChangedLogs.filter(log => log.resourceType == resourceType); + const resourceChangedLogs = allResourceChangedLogs.filter( + log => log.resourceType == resourceType, + ); const groupedLogs = SimLog.groupDuplicateTimestamps(resourceChangedLogs); - results[resourceType] = groupedLogs.map(logGroup => new ResourceChangedLogGroup( - { - raw: '', - logIndex: logGroup[0].logIndex, - timestamp: logGroup[0].timestamp, - source: logGroup[0].source, - target: logGroup[0].target, - actionId: null, - threat: 0, - }, - resourceType, - logGroup[0].valueBefore, - logGroup[logGroup.length - 1].valueAfter, - logGroup)); + results[resourceType] = groupedLogs.map( + logGroup => + new ResourceChangedLogGroup( + { + raw: '', + logIndex: logGroup[0].logIndex, + timestamp: logGroup[0].timestamp, + source: logGroup[0].source, + target: logGroup[0].target, + actionId: null, + threat: 0, + }, + resourceType, + logGroup[0].valueBefore, + logGroup[logGroup.length - 1].valueAfter, + logGroup, + ), + ); }); return results as Record>; @@ -734,16 +887,20 @@ export class MajorCooldownUsedLog extends SimLog { } toString(includeTimestamp = true): string { - return `${this.toStringPrefix(includeTimestamp)} Major cooldown used: ${this.newActionIdLink()}.`; + return `${this.toStringPrefix( + includeTimestamp, + )} Major cooldown used: ${this.newActionIdLink()}.`; } static parse(params: SimLogParams): Promise | null { const match = params.raw.match(/Major cooldown used: (.*)/); if (match) { - return ActionId.fromLogString(match[1]).fill(params.source?.index).then(cooldownId => { - params.actionId = cooldownId; - return new MajorCooldownUsedLog(params); - }); + return ActionId.fromLogString(match[1]) + .fill(params.source?.index) + .then(cooldownId => { + params.actionId = cooldownId; + return new MajorCooldownUsedLog(params); + }); } else { return null; } @@ -763,11 +920,17 @@ export class CastBeganLog extends SimLog { } toString(includeTimestamp = true): string { - return `${this.toStringPrefix(includeTimestamp)} Casting ${this.newActionIdLink()} (Cast time: ${this.castTime.toFixed(2)}s, Cost: ${this.manaCost.toFixed(1)} Mana).`; + return `${this.toStringPrefix( + includeTimestamp, + )} Casting ${this.newActionIdLink()} (Cast time: ${this.castTime.toFixed( + 2, + )}s, Cost: ${this.manaCost.toFixed(1)} Mana).`; } static parse(params: SimLogParams): Promise | null { - const match = params.raw.match(/Casting (.*) \(Cost = (\d+\.?\d*), Cast Time = (\d+\.?\d*)(m?s), Effective Time = (\d+\.?\d*)(m?s)\)/); + const match = params.raw.match( + /Casting (.*) \(Cost = (\d+\.?\d*), Cast Time = (\d+\.?\d*)(m?s), Effective Time = (\d+\.?\d*)(m?s)\)/, + ); if (match) { let castTime = parseFloat(match[3]); if (match[4] == 'ms') { @@ -777,10 +940,12 @@ export class CastBeganLog extends SimLog { if (match[6] == 'ms') { effectiveTime /= 1000; } - return ActionId.fromLogString(match[1]).fill(params.source?.index).then(castId => { - params.actionId = castId; - return new CastBeganLog(params, parseFloat(match[2]), castTime, effectiveTime); - }); + return ActionId.fromLogString(match[1]) + .fill(params.source?.index) + .then(castId => { + params.actionId = castId; + return new CastBeganLog(params, parseFloat(match[2]), castTime, effectiveTime); + }); } else { return null; } @@ -799,10 +964,12 @@ export class CastCompletedLog extends SimLog { static parse(params: SimLogParams): Promise | null { const match = params.raw.match(/Completed cast (.*)/); if (match) { - return ActionId.fromLogString(match[1]).fill(params.source?.index).then(castId => { - params.actionId = castId; - return new CastCompletedLog(params); - }); + return ActionId.fromLogString(match[1]) + .fill(params.source?.index) + .then(castId => { + params.actionId = castId; + return new CastCompletedLog(params); + }); } else { return null; } @@ -820,7 +987,11 @@ export class CastLog extends SimLog { // All instances of damage dealt from the completion of this cast until the completion of the next cast. readonly damageDealtLogs: Array; - constructor(castBeganLog: CastBeganLog, castCompletedLog: CastCompletedLog | null, damageDealtLogs: Array) { + constructor( + castBeganLog: CastBeganLog, + castCompletedLog: CastCompletedLog | null, + damageDealtLogs: Array, + ) { super({ raw: castBeganLog.raw, logIndex: castBeganLog.logIndex, @@ -837,13 +1008,16 @@ export class CastLog extends SimLog { this.damageDealtLogs = damageDealtLogs; if (this.castCompletedLog && this.castBeganLog) { - this.castTime = this.castCompletedLog.timestamp - this.castBeganLog.timestamp - this.effectiveTime = this.castCompletedLog.timestamp - this.castBeganLog.timestamp + this.castTime = this.castCompletedLog.timestamp - this.castBeganLog.timestamp; + this.effectiveTime = this.castCompletedLog.timestamp - this.castBeganLog.timestamp; } - if (this.castCompletedLog && this.damageDealtLogs.length == 1 && + if ( + this.castCompletedLog && + this.damageDealtLogs.length == 1 && this.castCompletedLog.timestamp < this.damageDealtLogs[0].timestamp && - !this.damageDealtLogs[0].tick) { + !this.damageDealtLogs[0].tick + ) { this.travelTime = this.damageDealtLogs[0].timestamp - this.castCompletedLog.timestamp; } else { this.travelTime = 0; @@ -851,7 +1025,9 @@ export class CastLog extends SimLog { } toString(includeTimestamp = true): string { - return `${this.toStringPrefix(includeTimestamp)} Casting ${this.actionId!.name} (Cast time = ${this.castTime.toFixed(2)}s).`; + return `${this.toStringPrefix(includeTimestamp)} Casting ${ + this.actionId!.name + } (Cast time = ${this.castTime.toFixed(2)}s).`; } totalDamage(): number { @@ -860,7 +1036,9 @@ export class CastLog extends SimLog { static fromLogs(logs: Array): Array { const castBeganLogs = logs.filter((log): log is CastBeganLog => log.isCastBegan()); - const castCompletedLogs = logs.filter((log): log is CastCompletedLog => log.isCastCompleted()); + const castCompletedLogs = logs.filter((log): log is CastCompletedLog => + log.isCastCompleted(), + ); const damageDealtLogs = logs.filter((log): log is DamageDealtLog => log.isDamageDealt()); const toBucketKey = (actionId: ActionId) => { @@ -873,7 +1051,9 @@ export class CastLog extends SimLog { } }; const castBeganLogsByAbility = bucket(castBeganLogs, log => toBucketKey(log.actionId!)); - const castCompletedLogsByAbility = bucket(castCompletedLogs, log => toBucketKey(log.actionId!)); + const castCompletedLogsByAbility = bucket(castCompletedLogs, log => + toBucketKey(log.actionId!), + ); const damageDealtLogsByAbility = bucket(damageDealtLogs, log => toBucketKey(log.actionId!)); const castLogs: Array = []; @@ -899,9 +1079,13 @@ export class CastLog extends SimLog { // Find all damage dealt logs between the cur and next cast completed logs. const ddLogs = []; - while (abilityDamageDealt && ddIdx < abilityDamageDealt.length && (!nextCcLog || abilityDamageDealt[ddIdx].timestamp < nextCcLog.timestamp)) { + while ( + abilityDamageDealt && + ddIdx < abilityDamageDealt.length && + (!nextCcLog || abilityDamageDealt[ddIdx].timestamp < nextCcLog.timestamp) + ) { ddLogs.push(abilityDamageDealt[ddIdx]); - ddIdx++ + ddIdx++; } castLogs.push(new CastLog(cbLog, ccLog, ddLogs)); } @@ -924,20 +1108,26 @@ export class StatChangeLog extends SimLog { toString(includeTimestamp = true): string { if (this.isGain) { - return `${this.toStringPrefix(includeTimestamp)} Gained ${this.stats} from ${this.newActionIdLink()}.`; + return `${this.toStringPrefix(includeTimestamp)} Gained ${ + this.stats + } from ${this.newActionIdLink()}.`; } else { - return `${this.toStringPrefix(includeTimestamp)} Lost ${this.stats} from fading ${this.newActionIdLink()}.`; + return `${this.toStringPrefix(includeTimestamp)} Lost ${ + this.stats + } from fading ${this.newActionIdLink()}.`; } } static parse(params: SimLogParams): Promise | null { const match = params.raw.match(/((Gained)|(Lost)) ({.*}) from (fading )?(.*)/); if (match) { - return ActionId.fromLogString(match[6]).fill(params.source?.index).then(effectId => { - params.actionId = effectId; - const sign = match[1] == 'Lost' ? -1 : 1; - return new StatChangeLog(params, sign == 1, match[4]); - }); + return ActionId.fromLogString(match[6]) + .fill(params.source?.index) + .then(effectId => { + params.actionId = effectId; + const sign = match[1] == 'Lost' ? -1 : 1; + return new StatChangeLog(params, sign == 1, match[4]); + }); } else { return null; } diff --git a/ui/scss/homepage/_homepage.scss b/ui/scss/homepage/_homepage.scss index 48d50167d1..02b9ba3f9d 100644 --- a/ui/scss/homepage/_homepage.scss +++ b/ui/scss/homepage/_homepage.scss @@ -1,189 +1,192 @@ -@use "sass:map"; +@use 'sass:map'; .homepage-image { - position: fixed; - width: 100%; - height: 100%; - background-image: url("/wotlk/assets/img/wotlk.jpg"); - background-repeat: no-repeat; - background-size: cover; - z-index: -1; + position: fixed; + width: 100%; + height: 100%; + background-image: url('/wotlk/assets/img/wotlk.jpg'); + background-repeat: no-repeat; + background-size: cover; + z-index: -1; } #homepage { - height: 100%; - display: flex; - flex-direction: column; - - - .homepage-header-container { - padding-top: calc(var(--container-padding) / 2); - } - - .homepage-content-container { - padding-top: var(--container-padding); - padding-bottom: var(--container-padding); - } - - .homepage-header { - .homepage-header-container { - display: flex; - - .navbar-brand-container { - display: flex; - - .wowsims-logo { - width: 6rem; - margin-right: map-get($spacers, 3); - } - - .wowsims-title { - @extend .display-1; - @extend .fw-bold; - color: $brand; - margin: 0; - line-height: 1; - } - } - - .homepage-header-collapse { - padding-top: map-get($spacers, 3); - padding-bottom: map-get($spacers, 3); - align-items: flex-end; - justify-content: flex-end; - } - } - - .expansion-title { - color: $wrath-blue; - margin: 0; - } - } - - .homepage-content-container { - flex-direction: column; - - .info-container { - margin-bottom: var(--container-padding); - display: flex; - flex-direction: column; - - .wowsims-info { - @extend .fs-4; - width: 75%; - margin: 0; - } - } - - .sim-links-container { - display: flex; - flex-direction: column; - - .sim-links { - display: flex; - flex-wrap: wrap; - - &:not(:last-child) { - margin-bottom: map-get($spacers, 5); - } - - &> .sim-link, &> .sim-link-dropdown { - max-width: 25%; - flex: 1 25%; - margin-right: 25%; - } - - .raid-sim-link { - flex-grow: 0 !important; - } - - .sim-link { - .sim-link-content { - .sim-link-label:first-of-type { - font-size: 1rem; - } - - .sim-link-label:last-of-type { - font-size: 1.75rem; - } - } - } - } - } - } + height: 100%; + display: flex; + flex-direction: column; + + .homepage-header-container { + padding-top: calc(var(--container-padding) / 2); + } + + .homepage-content-container { + padding-top: var(--container-padding); + padding-bottom: var(--container-padding); + } + + .homepage-header { + .homepage-header-container { + display: flex; + + .navbar-brand-container { + display: flex; + + .wowsims-logo { + width: 6rem; + margin-right: map-get($spacers, 3); + } + + .wowsims-title { + @extend .display-1; + @extend .fw-bold; + color: $brand; + margin: 0; + line-height: 1; + } + } + + .homepage-header-collapse { + padding-top: map-get($spacers, 3); + padding-bottom: map-get($spacers, 3); + align-items: flex-end; + justify-content: flex-end; + } + } + + .expansion-title { + color: $wrath-blue; + margin: 0; + } + } + + .homepage-content-container { + flex-direction: column; + + .info-container { + margin-bottom: var(--container-padding); + display: flex; + flex-direction: column; + + .wowsims-info { + @extend .fs-4; + width: 75%; + margin: 0; + } + } + + .sim-links-container { + display: flex; + flex-direction: column; + + .sim-links { + display: flex; + flex-wrap: wrap; + + &:not(:last-child) { + margin-bottom: map-get($spacers, 5); + } + + & > .sim-link, + & > .sim-link-dropdown { + max-width: 25%; + flex: 1 25%; + margin-right: 25%; + } + + .raid-sim-link { + flex-grow: 0 !important; + } + + .sim-link { + .sim-link-content { + .sim-link-label:first-of-type { + font-size: 1rem; + } + + .sim-link-label:last-of-type { + font-size: 1.75rem; + } + } + } + } + } + } } @include media-breakpoint-down(lg) { - #homepage { - .homepage-content-container { - .info-container { - .wowsims-info { - width: 100%; - } - } - - .sim-links-container { - margin-left: map-get($spacers, 3) * -1; - margin-right: map-get($spacers, 3) * -1; - - .sim-links { - margin-bottom: 0 !important; - - .sim-link, .sim-link-dropdown { - flex: 1 50%; - margin-right: 0 !important; - } - - .sim-link-dropdown { - .dropdown-menu { - position: relative !important; - inset: 0 !important; - transform: none !important; - } - } - } - } - } - } + #homepage { + .homepage-content-container { + .info-container { + .wowsims-info { + width: 100%; + } + } + + .sim-links-container { + margin-left: map-get($spacers, 3) * -1; + margin-right: map-get($spacers, 3) * -1; + + .sim-links { + margin-bottom: 0 !important; + + .sim-link, + .sim-link-dropdown { + max-width: 50%; + flex: 1 50%; + margin-right: 0 !important; + } + + .sim-link-dropdown { + .dropdown-menu { + position: relative !important; + inset: 0 !important; + transform: none !important; + } + } + } + } + } + } } @include media-breakpoint-down(md) { - #homepage { - .homepage-header-container, - .homepage-content-container, - .homepage-footer-container { - padding-top: map-get($spacers, 3); - padding-bottom: map-get($spacers, 3); - } - - .homepage-header { - .homepage-header-container { - display: flex; - - .navbar-brand-container { - width: 100%; - justify-content: space-between; - align-items: flex-end; - - .wowsims-logo { - width: 48px; - } - } - } - } - - .homepage-content-container { - .info-container { - margin-bottom: map-get($spacers, 3); - } - - .sim-links-container { - .sim-links { - .sim-link, .sim-link-dropdown { - flex: 1 100% !important; - } - } - } - } - } + #homepage { + .homepage-header-container, + .homepage-content-container, + .homepage-footer-container { + padding-top: map-get($spacers, 3); + padding-bottom: map-get($spacers, 3); + } + + .homepage-header { + .homepage-header-container { + display: flex; + + .navbar-brand-container { + width: 100%; + justify-content: space-between; + align-items: flex-end; + + .wowsims-logo { + width: 48px; + } + } + } + } + + .homepage-content-container { + .info-container { + margin-bottom: map-get($spacers, 3); + } + + .sim-links-container { + .sim-links { + .sim-link, + .sim-link-dropdown { + flex: 1 100% !important; + } + } + } + } + } }