/* globals $ */ var smapi = smapi || {}; var app; var messages; // Necessary helper method for updating our text filter in a performant way. // Wouldn't want to update it for every individually typed character. function debounce(fn, delay) { var timeoutID = null return function () { clearTimeout(timeoutID) var args = arguments var that = this timeoutID = setTimeout(function () { fn.apply(that, args) }, delay) } } // Case insensitive text searching and match word searching is best done in // regex, so if the user isn't trying to use regex, escape their input. function escapeRegex(text) { return text.replace(/[.*+?^${}()|[\]\\]/g, '\\$&'); } // Use a scroll event to apply a sticky effect to the filters / pagination // bar. We can't just use "position: sticky" due to how the page is structured // but this works well enough. $(function () { let sticking = false; document.addEventListener('scroll', function (event) { const filters = document.getElementById('filters'); const holder = document.getElementById('filterHolder'); if (!filters || !holder) return; const offset = holder.offsetTop; const should_stick = window.pageYOffset > offset; if (should_stick === sticking) return; sticking = should_stick; if (sticking) { holder.style.marginBottom = `calc(1em + ${filters.offsetHeight}px)`; filters.classList.add('sticky'); } else { filters.classList.remove('sticky'); holder.style.marginBottom = ''; } }); }); // This method is called when we click a log line to toggle the visibility // of a section. Binding methods is problematic with functional components // so we just use the `data-section` parameter and our global reference // to the app. smapi.clickLogLine = function (event) { app.toggleSection(event.currentTarget.dataset.section); event.preventDefault(); return false; } // And these methods are called when doing pagination. Just makes things // easier, so may as well use helpers. smapi.prevPage = function () { app.prevPage(); } smapi.nextPage = function () { app.nextPage(); } smapi.changePage = function (event) { if (typeof event === 'number') app.changePage(event); else if (event) { const page = parseInt(event.currentTarget.dataset.page); if (!isNaN(page) && isFinite(page)) app.changePage(page); } } smapi.logParser = function (data, sectionUrl) { if (!data) data = {}; // internal filter counts var stats = data.stats = { modsShown: 0, modsHidden: 0 }; function updateModFilters() { // counts stats.modsShown = 0; stats.modsHidden = 0; for (var key in data.showMods) { if (data.showMods.hasOwnProperty(key)) { if (data.showMods[key]) stats.modsShown++; else stats.modsHidden++; } } } // load our data // Rather than pre-rendering the list elements into the document, we read // a lot of JSON and use Vue to build the list. This is a lot more // performant and easier on memory.Our JSON is stored in special script // tags, that we later remove to let the browser clean up even more memory. let nodeParsedMessages = document.querySelector('script#parsedMessages'); if (nodeParsedMessages) { messages = JSON.parse(nodeParsedMessages.textContent) || []; const logLevels = JSON.parse(document.querySelector('script#logLevels').textContent) || {}; const logSections = JSON.parse(document.querySelector('script#logSections').textContent) || {}; const modSlugs = JSON.parse(document.querySelector('script#modSlugs').textContent) || {}; // Remove all references to the script tags and remove them from the // DOM so that the browser can clean them up. nodeParsedMessages.remove(); document.querySelector('script#logLevels').remove(); document.querySelector('script#logSections').remove(); document.querySelector('script#modSlugs').remove(); nodeParsedMessages = null; // Pre-process the messages since they aren't quite serialized in // the format we want. We also want to freeze every last message // so that Vue won't install its change listening behavior. for (let i = 0, length = messages.length; i < length; i++) { const msg = messages[i]; msg.id = i; msg.LevelName = logLevels && logLevels[msg.Level]; msg.SectionName = logSections && logSections[msg.Section]; msg.ModSlug = modSlugs && modSlugs[msg.Mod] || msg.Mod; // For repeated messages, since our component // can't return two rows, just insert a second message // which will display as the message repeated notice. if (msg.Repeated > 0 && ! msg.isRepeated) { const second = { id: i + 1, Level: msg.Level, Section: msg.Section, Mod: msg.Mod, Repeated: msg.Repeated, isRepeated: true, }; messages.splice(i + 1, 0, second); length++; } Object.freeze(msg); } Object.freeze(messages); } else messages = []; // set local time started if (data.logStarted) data.localTimeStarted = ("0" + data.logStarted.getHours()).slice(-2) + ":" + ("0" + data.logStarted.getMinutes()).slice(-2); // Add some properties to the data we're passing to Vue. data.totalMessages = messages.length; data.filterText = ''; data.filterRegex = ''; data.showContentPacks = true; data.useHighlight = true; data.useRegex = false; data.useInsensitive = true; data.useWord = false; data.perPage = 1000; data.page = 1; // Now load these values. if (localStorage.settings) { try { const saved = JSON.parse(localStorage.settings); if (saved.hasOwnProperty('showContentPacks')) data.showContentPacks = saved.showContentPacks; if (saved.hasOwnProperty('useHighlight')) dat.useHighlight = saved.useHighlight; if (saved.hasOwnProperty('useRegex')) data.useRegex = saved.useRegex; if (saved.hasOwnProperty('useInsensitive')) data.useInsensitive = saved.useInsensitive; if (saved.hasOwnProperty('useWord')) data.useWord = saved.useWord; } catch { /* ignore errors */ } } // This would be easier if we could just use JSX but this project doesn't // have a proper JavaScript build environment and I really don't feel // like setting one up. // Add a number formatter so that our numbers look nicer. const fmt = window.Intl && Intl.NumberFormat && new Intl.NumberFormat(); function formatNumber(value) { if (!fmt || !fmt.format) return `${value}`; return fmt.format(value); } Vue.filter('number', formatNumber); // Strictly speaking, we don't need this. However, due to the way our // Vue template is living in-page the browser is "helpful" and moves // our s outside of a basic since obviously they // aren't table rows and don't belong inside a table. By using another // Vue component, we avoid that. Vue.component('log-table', { functional: true, render: function (createElement, context) { return createElement('table', { attrs: { id: 'log' } }, context.children); } }); // The component draws a nice message under the filters // telling a user how many messages match their filters, and also expands // on how many of them they're seeing because of pagination. Vue.component('filter-stats', { functional: true, render: function (createElement, context) { const props = context.props; if (props.pages > 1) return createElement('div', { class: 'stats' }, [ 'showing ', createElement('strong', formatNumber(props.start + 1)), ' to ', createElement('strong', formatNumber(props.end)), ' of ', createElement('strong', formatNumber(props.filtered)), ' (total: ', createElement('strong', formatNumber(props.total)), ')' ]); return createElement('div', { class: 'stats' }, [ 'showing ', createElement('strong', formatNumber(props.filtered)), ' out of ', createElement('strong', formatNumber(props.total)) ]); } }); // Next up we have which renders the pagination list. This has a // helper method to make building the list of links easier. function addPageLink(page, links, visited, createElement, currentPage) { if (visited.has(page)) return; if (page > 1 && !visited.has(page - 1)) links.push(' … '); visited.add(page); links.push(createElement('span', { class: page == currentPage ? 'active' : null, attrs: { 'data-page': page }, on: { click: smapi.changePage } }, formatNumber(page))); } Vue.component('pager', { functional: true, render: function (createElement, context) { const props = context.props; if (props.pages <= 1) return null; const visited = new Set; const pageLinks = []; for (let i = 1; i <= 2; i++) addPageLink(i, pageLinks, visited, createElement, props.page); for (let i = props.page - 2; i <= props.page + 2; i++) { if (i < 1 || i > props.pages) continue; addPageLink(i, pageLinks, visited, createElement, props.page); } for (let i = props.pages - 2; i <= props.pages; i++) { if (i < 1) continue; addPageLink(i, pageLinks, visited, createElement, props.page); } return createElement('div', { class: 'pager' }, [ createElement('span', { class: props.page <= 1 ? 'disabled' : null, on: { click: smapi.prevPage } }, 'Prev'), ' ', 'Page ', formatNumber(props.page), ' of ', formatNumber(props.pages), ' ', createElement('span', { class: props.page >= props.pages ? 'disabled' : null, on: { click: smapi.nextPage } }, 'Next'), createElement('div', {}, pageLinks) ]); } }); // Our functional component draws each log line. Vue.component('log-line', { functional: true, props: { showScreenId: { type: Boolean, required: true }, message: { type: Object, required: true }, sectionExpanded: { type: Boolean, required: false }, highlight: { type: Boolean, required: false } }, render: function (createElement, context) { const msg = context.props.message; const level = msg.LevelName; if (msg.isRepeated) return createElement('tr', { class: [ "mod", level, "mod-repeat" ] }, [ createElement('td', { attrs: { colspan: context.props.showScreenId ? 4 : 3 } }, ''), createElement('td', `repeats ${msg.Repeated} times`) ]); const events = {}; let toggleMessage; if (msg.IsStartOfSection) { const visible = context.props.sectionExpanded; events.click = smapi.clickLogLine; toggleMessage = visible ? 'This section is shown. Click here to hide it.' : 'This section is hidden. Click here to show it.'; } let text = msg.Text; const filter = window.app && app.filterRegex; if (text && filter && context.props.highlight) { text = []; let match, consumed = 0, idx = 0; filter.lastIndex = -1; // Our logic to highlight the text is a bit funky because we // want to group consecutive matches to avoid a situation // where a ton of single characters are in their own elements // if the user gives us bad input. while (match = filter.exec(msg.Text)) { // Do we have an area of non-matching text? This // happens if the new match's index is further // along than the last index. if (match.index > idx) { // Alright, do we have a previous match? If // we do, we need to consume some text. if (consumed < idx) text.push(createElement('strong', {}, msg.Text.slice(consumed, idx))); text.push(msg.Text.slice(idx, match.index)); consumed = match.index; } idx = match.index + match[0].length; } // Add any trailing text after the last match was found. if (consumed < msg.Text.length) { if (consumed < idx) text.push(createElement('strong', {}, msg.Text.slice(consumed, idx))); if (idx < msg.Text.length) text.push(msg.Text.slice(idx)); } } return createElement('tr', { class: [ "mod", level, msg.IsStartOfSection ? "section-start" : null ], attrs: { 'data-section': msg.SectionName }, on: events }, [ createElement('td', msg.Time), context.props.showScreenId ? createElement('td', msg.ScreenId) : null, createElement('td', level.toUpperCase()), createElement('td', { attrs: { 'data-title': msg.Mod } }, msg.Mod), createElement('td', [ createElement('span', { class: 'log-message-text' }, text), msg.IsStartOfSection ? createElement('span', { class: 'section-toggle-message' }, [ ' ', toggleMessage ]) : null ]) ]); } }); // init app app = new Vue({ el: '#output', data: data, computed: { anyModsHidden: function () { return stats.modsHidden > 0; }, anyModsShown: function () { return stats.modsShown > 0; }, showScreenId: function () { return this.screenIds.length > 1; }, // Maybe not strictly necessary, but the Vue template is being // weird about accessing data entries on the app rather than // computed properties. visibleSections: function () { const ret = []; for (const [k, v] of Object.entries(this.showSections)) if (v !== false) ret.push(k); return ret; }, hideContentPacks: function () { return !data.showContentPacks; }, // Filter messages for visibility. filterUseRegex: function () { return data.useRegex; }, filterInsensitive: function () { return data.useInsensitive; }, filterUseWord: function () { return data.useWord; }, shouldHighlight: function () { return data.useHighlight; }, filteredMessages: function () { if (!messages) return []; const start = performance.now(); const ret = []; // This is slightly faster than messages.filter(), which is // important when working with absolutely huge logs. for (let i = 0, length = messages.length; i < length; i++) { const msg = messages[i]; if (msg.SectionName && !msg.IsStartOfSection && !this.sectionsAllow(msg.SectionName)) continue; if (!this.filtersAllow(msg.ModSlug, msg.LevelName)) continue; let text = msg.Text || (i > 0 ? messages[i - 1].Text : null); if (this.filterRegex) { this.filterRegex.lastIndex = -1; if (!text || !this.filterRegex.test(text)) continue; } else if (this.filterText && (!text || text.indexOf(this.filterText) == -1)) continue; ret.push(msg); } const end = performance.now(); console.log(`filter took ${end - start}ms`); return ret; }, // And the rest are about pagination. start: function () { return (this.page - 1) * data.perPage; }, end: function () { return this.start + this.visibleMessages.length; }, totalPages: function () { return Math.ceil(this.filteredMessages.length / data.perPage); }, // visibleMessages: function () { if (this.totalPages <= 1) return this.filteredMessages; const start = this.start; const end = start + data.perPage; return this.filteredMessages.slice(start, end); } }, created: function () { this.loadFromUrl = this.loadFromUrl.bind(this); window.addEventListener('popstate', this.loadFromUrl); this.loadFromUrl(); }, methods: { // Mostly I wanted people to know they can override the PerPage // message count with a URL parameter, but we can read Page too. // In the future maybe we should read *all* filter state so a // user can link to their exact page state for someone else? loadFromUrl: function () { const params = new URL(location).searchParams; if (params.has('PerPage')) try { const perPage = parseInt(params.get('PerPage')); if (!isNaN(perPage) && isFinite(perPage) && perPage > 0) data.perPage = perPage; } catch { /* ignore errors */ } if (params.has('Page')) try { const page = parseInt(params.get('Page')); if (!isNaN(page) && isFinite(page) && page > 0) this.page = page; } catch { /* ignore errors */ } }, toggleLevel: function (id) { if (!data.enableFilters) return; this.showLevels[id] = !this.showLevels[id]; }, toggleContentPacks: function () { data.showContentPacks = !data.showContentPacks; this.saveSettings(); }, toggleFilterUseRegex: function () { data.useRegex = !data.useRegex; this.saveSettings(); this.updateFilterText(); }, toggleFilterInsensitive: function () { data.useInsensitive = !data.useInsensitive; this.saveSettings(); this.updateFilterText(); }, toggleFilterWord: function () { data.useWord = !data.useWord; this.saveSettings(); this.updateFilterText(); }, toggleHighlight: function () { data.useHighlight = !data.useHighlight; this.saveSettings(); }, prevPage: function () { if (this.page <= 1) return; this.page--; this.updateUrl(); }, nextPage: function () { if (this.page >= this.totalPages) return; this.page++; this.updateUrl(); }, changePage: function (page) { if (page < 1 || page > this.totalPages) return; this.page = page; this.updateUrl(); }, // Persist settings into localStorage for use the next time // the user opens a log. saveSettings: function () { localStorage.settings = JSON.stringify({ showContentPacks: data.showContentPacks, useRegex: data.useRegex, useInsensitive: data.useInsensitive, useWord: data.useWord, useHighlight: data.useHighlight }); }, // Whenever the page is changed, replace the current page URL. Using // replaceState rather than pushState to avoid filling the tab history // with tons of useless history steps the user probably doesn't // really care about. updateUrl: function () { const url = new URL(location); url.searchParams.set('Page', data.page); url.searchParams.set('PerPage', data.perPage); window.history.replaceState(null, document.title, url.toString()); }, // We don't want to update the filter text often, so use a debounce with // a quarter second delay. We basically always build a regular expression // since we use it for highlighting, and it also make case insensitivity // much easier. updateFilterText: debounce(function () { let text = this.filterText = document.querySelector('input[type=text]').value; if (!text || !text.length) { this.filterText = ''; this.filterRegex = null; } else { if (!data.useRegex) text = escapeRegex(text); this.filterRegex = new RegExp( data.useWord ? `\\b${text}\\b` : text, data.useInsensitive ? 'ig' : 'g' ); } }, 250), toggleMod: function (id) { if (!data.enableFilters) return; var curShown = this.showMods[id]; // first filter: only show this by default if (stats.modsHidden === 0) { this.hideAllMods(); this.showMods[id] = true; } // unchecked last filter: reset else if (stats.modsShown === 1 && curShown) this.showAllMods(); // else toggle else this.showMods[id] = !this.showMods[id]; updateModFilters(); }, toggleSection: function (name) { if (!data.enableFilters) return; this.showSections[name] = !this.showSections[name]; }, showAllMods: function () { if (!data.enableFilters) return; for (var key in this.showMods) { if (this.showMods.hasOwnProperty(key)) { this.showMods[key] = true; } } updateModFilters(); }, hideAllMods: function () { if (!data.enableFilters) return; for (var key in this.showMods) { if (this.showMods.hasOwnProperty(key)) { this.showMods[key] = false; } } updateModFilters(); }, filtersAllow: function (modId, level) { return this.showMods[modId] !== false && this.showLevels[level] !== false; }, sectionsAllow: function (section) { return this.showSections[section] !== false; } } }); /********** ** Upload form *********/ var input = $("#input"); if (input.length) { // file upload smapi.fileUpload({ chooseFileLink: $("#choose-file-link"), chooseFileInput: $("#inputFile"), contentArea: input, submitButton: $("#submit") }); } };