Gisle Aune
5 years ago
6 changed files with 195 additions and 0 deletions
-
93marko/page/logs-content/components/log-suggestions/component.js
-
41marko/page/logs-content/components/log-suggestions/index.marko
-
46marko/page/logs-content/components/log-suggestions/style.less
-
2marko/page/logs-content/components/page/index.marko
-
4marko/page/logs-content/components/page/style.less
-
9rpdata/api/LogHeader.js
@ -0,0 +1,93 @@ |
|||||
|
const {logHeaderApi} = require("../../../../../rpdata/api/LogHeader") |
||||
|
|
||||
|
module.exports = class { |
||||
|
onCreate(input) { |
||||
|
this.state = { |
||||
|
id: input.log.id, |
||||
|
channelName: input.log.channelName, |
||||
|
eventName: input.log.eventName, |
||||
|
nexts: [], |
||||
|
prevs: [], |
||||
|
loaded: false, |
||||
|
error: null, |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
onMount() { |
||||
|
this.timeout = setTimeout(() => { |
||||
|
logHeaderApi.list().then(logs => { |
||||
|
const index = logs.findIndex(l => l.id === this.state.id); |
||||
|
if (index === -1) { |
||||
|
return Promise.reject("This log not found in list.") |
||||
|
} |
||||
|
const log = logs[index] |
||||
|
const characterIds = log.characters.map(c => c.id); |
||||
|
const nexts = [] |
||||
|
const prevs = [] |
||||
|
const foundGroups = {} |
||||
|
let foundChannel = false |
||||
|
|
||||
|
for (let i = index - 1; i >= 0; --i) { |
||||
|
const log2 = logs[i] |
||||
|
const hasEvent = (log.eventName ? log.eventName === log2.eventName : false) |
||||
|
const hasChannel = (log.channelName === log2.channelName) |
||||
|
|
||||
|
const characters = log2.findCharacters(...characterIds) |
||||
|
const characterIdsKey = characters.map(c => c.id).sort().join(",") |
||||
|
|
||||
|
const suggestion = {log: log2, characters, hasEvent, hasChannel} |
||||
|
|
||||
|
if (hasChannel && !foundChannel) { |
||||
|
foundChannel = true |
||||
|
foundGroups[characterIdsKey] = true |
||||
|
|
||||
|
nexts.push(suggestion) |
||||
|
} else if (hasEvent) { |
||||
|
foundGroups[characterIdsKey] = true |
||||
|
|
||||
|
nexts.push(suggestion) |
||||
|
} else if (nexts.length < 8 && !hasEvent && characters.length > 1 && !foundGroups[characterIdsKey]) { |
||||
|
foundGroups[characterIdsKey] = true |
||||
|
nexts.push(suggestion) |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
for (let i = index + 1; i < logs.length; ++i) { |
||||
|
const log2 = logs[i] |
||||
|
const hasEvent = (log.eventName ? log.eventName === log2.eventName : false) |
||||
|
const hasChannel = (log.channelName === log2.channelName) |
||||
|
|
||||
|
const characters = log2.findCharacters(...characterIds) |
||||
|
const characterIdsKey = characters.map(c => c.id).sort().join(",") |
||||
|
|
||||
|
const suggestion = {log: log2, characters, hasEvent, hasChannel} |
||||
|
|
||||
|
if (hasChannel && !foundChannel) { |
||||
|
foundChannel = true |
||||
|
foundGroups[characterIdsKey] = true |
||||
|
|
||||
|
prevs.push(suggestion) |
||||
|
} else if (hasEvent) { |
||||
|
foundGroups[characterIdsKey] = true |
||||
|
|
||||
|
prevs.push(suggestion) |
||||
|
} else if (prevs.length < 8 && !hasEvent && characters.length > 1 && !foundGroups[characterIdsKey]) { |
||||
|
foundGroups[characterIdsKey] = true |
||||
|
prevs.push(suggestion) |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
this.state.nexts = nexts; |
||||
|
this.state.prevs = prevs; |
||||
|
this.state.loaded = true; |
||||
|
}).catch(err => { |
||||
|
console.warn("Suggestions load error:", err) |
||||
|
this.suggestionsError = "Failed to load suggestions: " + err.toString() |
||||
|
}) |
||||
|
}, 1000) |
||||
|
} |
||||
|
|
||||
|
onUnmount() { |
||||
|
clearTimeout(this.timeout) |
||||
|
} |
||||
|
} |
@ -0,0 +1,41 @@ |
|||||
|
import moment from "moment" |
||||
|
|
||||
|
<div class="log-suggestions"> |
||||
|
<if (state.loaded && state.error == null)> |
||||
|
<if (state.nexts.length > 0 || state.prevs.length > 0)> |
||||
|
<if (state.nexts.length > 0)> |
||||
|
<h2 class="color-primary">Next Logs</h2> |
||||
|
<div for(s in state.nexts) class="suggestion color-primary"> |
||||
|
<div class="suggestion-header color-primary"><a href=`/logs/${s.log.id}`>${s.log.name || `${s.log.channelName} - ${moment(s.log.date).format("MMMM D, YYYY")}`}</a></div> |
||||
|
<div for(paragraph in (s.log.description||"").split("\n").filter(p => p)) class="suggestion-description color-text">${paragraph}</div> |
||||
|
<div class="suggestion-tags"> |
||||
|
<div if(s.hasEvent) class="suggestion-tag color-tag-event">${s.log.eventName}</div> |
||||
|
<div if(s.hasChannel) class="suggestion-tag color-tag-location">${s.log.channelName}</div> |
||||
|
<div for(c in s.characters) class="suggestion-tag color-tag-character">${c.name}</div> |
||||
|
</div> |
||||
|
</div> |
||||
|
</if> |
||||
|
<if (state.prevs.length > 0)> |
||||
|
<h2 class="color-primary">Previous Logs</h2> |
||||
|
<div for(s in state.prevs) class="suggestion color-primary"> |
||||
|
<div class="suggestion-header color-primary"><a href=`/logs/${s.log.id}`>${s.log.name || `${s.log.channelName} - ${moment(s.log.date).format("MMMM D, YYYY")}`}</a></div> |
||||
|
<div for(paragraph in (s.log.description||"").split("\n").filter(p => p)) class="suggestion-description color-text">${paragraph}</div> |
||||
|
<div class="suggestion-tags"> |
||||
|
<div if(s.hasEvent) class="suggestion-tag color-tag-event">${s.log.eventName}</div> |
||||
|
<div if(s.hasChannel) class="suggestion-tag color-tag-location">${s.log.channelName}</div> |
||||
|
<div for(c in s.characters) class="suggestion-tag color-tag-character">${c.name}</div> |
||||
|
</div> |
||||
|
</div> |
||||
|
</if> |
||||
|
</if> |
||||
|
<else> |
||||
|
<div class="empty color-menu">(No suggestions)</div> |
||||
|
</else> |
||||
|
</if> |
||||
|
<else-if(state.error != null)> |
||||
|
<div class="error color-danger">{state.error}</div> |
||||
|
</else-if> |
||||
|
<else> |
||||
|
<div class="loading color-menu">Loading suggestions...</div> |
||||
|
</else> |
||||
|
</div> |
@ -0,0 +1,46 @@ |
|||||
|
div.log-suggestions { |
||||
|
padding: 0 0.5ch; |
||||
|
margin-top: 1em; |
||||
|
margin-bottom: 2em; |
||||
|
|
||||
|
h2 { |
||||
|
text-align: center; |
||||
|
margin: 0; |
||||
|
margin-bottom: 0.25em; |
||||
|
} |
||||
|
|
||||
|
div.empty, div.loading { |
||||
|
text-align: center; |
||||
|
} |
||||
|
|
||||
|
div.suggestion { |
||||
|
margin-bottom: 0.5em; |
||||
|
padding: 0.5em; |
||||
|
|
||||
|
div.suggestion-header { |
||||
|
margin: 0.125em 0.5ch; |
||||
|
|
||||
|
a { |
||||
|
color: inherit; |
||||
|
} |
||||
|
a:hover { |
||||
|
color: white; |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
div.suggestion-description { |
||||
|
margin: 0.125em 0.5ch; |
||||
|
} |
||||
|
|
||||
|
div.suggestion-tags { |
||||
|
margin-top: 0.25em; |
||||
|
|
||||
|
div.suggestion-tag { |
||||
|
display: inline-block; |
||||
|
padding: 0 0.5ch; |
||||
|
margin: 0.125em 0.5ch; |
||||
|
border: 0.25px solid; |
||||
|
} |
||||
|
} |
||||
|
} |
||||
|
} |
Write
Preview
Loading…
Cancel
Save
Reference in new issue