Display map details on chunithm-net
// ==UserScript==
// @name chuni-net - Map Details
// @namespace esterTion
// @license MIT
// @match https://chunithm-net-eng.com/mobile/record/
// @match https://new.chunithm-net.com/mobile/record/
// @match https://chunithm.wahlap.com/mobile/record/
// @grant GM.xmlHttpRequest
// @version 1.3.4
// @author esterTion
// @description Display map details on chunithm-net
// @run-at document-end
// ==/UserScript==
const host = location.hostname
const server = host === 'new.chunithm-net.com' ? 'jp' : host === 'chunithm-net-eng.com' ? 'ex' : host === 'chunithm.wahlap.com' ? 'cn' : ''
if (!server) throw new Error('unknown server')
const imageBase = server === 'jp' ? '/chuni-mobile/html/mobile/images' : '/mobile/images'
const PAGE_DATA_IS_SHRINKED = server !== 'cn'
// createElement
function _(e,t,i){var a=null;if("text"===e)return document.createTextNode(t);a=document.createElement(e);for(var n in t)if("style"===n)for(var o in t.style)a.style[o]=t.style[o];else if("className"===n)a.className=t[n];else if("event"===n)for(var o in t.event)a.addEventListener(o,t.event[o]);else a.setAttribute(n,t[n]);if(i)if("string"==typeof i)a.innerHTML=i;else if(Array.isArray(i))for(var l=0;l<i.length;l++)null!=i[l]&&a.appendChild(i[l]);return a}
function sleep(t){return new Promise(function(e){setTimeout(e,t)})}
const MAP_AREA_TASK = ''
const MAP_AREA_BOOST = ''
const MAP_AREA_HIDDEN_LOCK = ''
const localStorageTimeKey = 'CNMD_maps_info_time'
const localStorageDataKey = 'CNMD_maps_info'
let mapInfo = []
function loadLocalInfo() {
if (!localStorage[localStorageDataKey]) return
mapInfo = JSON.parse(localStorage[localStorageDataKey])
}
function checkUpdateForLocalInfo() {
const today = getDateStringForUpdate()
if (!localStorage[localStorageTimeKey] || localStorage[localStorageTimeKey] !== today) {
downloadInfo(today)
}
}
async function downloadInfo(today) {
console.log('downloading map info')
switch (server) {
case 'jp': {
throw new Error('not implemented')
break;
}
case 'ex': {
await fetchJson('https://estertion.win/__private__/chuni-intl-maps.json').then(r => mapInfo = r)
break;
}
case 'cn': {
await fetchJson('https://estertion.win/__private__/chuni-chn-maps.json').then(r => mapInfo = r)
break;
}
}
localStorage[localStorageDataKey] = JSON.stringify(mapInfo)
localStorage[localStorageTimeKey] = today
console.log('stored map info: ', Object.keys(mapInfo).length, 'entries')
addRemainingMaps()
}
function getDateStringForUpdate() {
const d = new Date
d.setTime(d.getTime() + d.getTimezoneOffset() * 60e3 + {jp:11,ex:11,cn:10}[server]*3600e3)
return [d.getUTCFullYear(), d.getUTCMonth()+1, d.getUTCDate()].join('/')
}
function fetchJson(url) {
return new Promise((res, rej) => {
GM.xmlHttpRequest({
url: url + '?_=' + Date.now(),
responseType: 'json',
method: 'GET',
onload: r => res(r.response),
onerror: e => rej(e),
})
})
}
function getHiddenType(type, text = '地图', tag = 'div') {
if (type === 1) {
return _(tag, { className: 'hidden_type_desc' }, [_('text', `(达成特定条件才会解锁的${text})`)])
}
if (type === 2) {
return _(tag, { className: 'hidden_type_desc' }, [_('text', `(达成特定条件才会显示的${text})`)])
}
return new Comment('hidden type')
}
const mapBlockMap = {}
function generatePage(map, page) {
const mapTitle = map.name
const isInfinite = mapTitle.indexOf('ep. ∞') !== -1
const currentPage = page
const totalPage = isInfinite ? Infinity : map.areas.slice(-1)[0].page + 1
let allClearedCheckbox
const mapPage = _('div', { className: 'map_block w400' }, [
_('div', { className: ['map-paginator', page>1?'has-prev':'', page<totalPage?'has-next':''].join(' ') }, [
_('span', { className: 'prev', event: { click: _ => { mapPage.parentNode.replaceChild(getPage(mapTitle, page - 1), mapPage); executeMarquee() } } }, [_('text', '<')]),
_('span', { className: 'next', event: { click: _ => { mapPage.parentNode.replaceChild(getPage(mapTitle, page + 1), mapPage); executeMarquee() } } }, [_('text', '>')]),
]),
_('div', { className: 'map_title w388' }, [
_('div', { className: 'map_title_text text_l text_b' }, [
_('text', mapTitle),
]),
_('div', { className: 'map_titale_page' }, [
_('div', { className: 'map_title_page_num font_90' }, [
_('text', currentPage),
]),
isInfinite
? _('div', { className: 'map_title_page_infinit' }, [_('img', { src: `${imageBase}/progress_RewardIconRemainBlock_Infinity.png`})])
: _('div', { className: 'map_title_page_den font_90' }, [_('text', totalPage),]),
]),
]),
_('div', { className: 'maparea_period w388' }, [
_('label', {}, [
allClearedCheckbox = _('input', { type: 'checkbox', event: { change: e => {
storeMapPageProgress(mapTitle, currentPage)
storeMapPageProgress(mapTitle, currentPage + (e.target.checked?1:-1))
mapPage.parentNode.replaceChild(getPage(mapTitle, page), mapPage); executeMarquee()
}}}),
_('text', '区域全部完成'),
]),
getHiddenType(map.hidden)
]),
])
const pageAreas = new Array(9)
map.areas.forEach(area => {
if (area.page !== page - 1) return
pageAreas[area.position] = area
})
if (isInfinite && page > 3) {
const area = map.areas.slice(-1)[0]
pageAreas[area.position] = area
}
let areaCount = 0, areaCleared = 0
for (let i = 0; i < 9; i++) {
if (!pageAreas[i]) {
mapPage.appendChild(_('div', { className: 'maparea_block'}, [
_('div', {className: 'maparea_blank'})
]))
} else {
const index = i
const areaLength = pageAreas[i].rewards.slice(-1)[0].position
const progress = getMapTileProgress(mapTitle, currentPage, index)
const isCleared = progress.finished
const areaLengthText = isCleared ? 0 : (areaLength - progress.position)
const pageAreasCount = pageAreas.filter(a => a).length
const tile = mapPage.appendChild(_('div', { className: 'maparea_block user_data_friend_tap', event: { click: () => showMapAreaDetail(mapTitle, currentPage, index, isCleared ? pageAreasCount : 0, areaLengthText) } }, [
getAreaTile(pageAreas[i], isCleared, areaLengthText, isCleared ? pageAreasCount : 0)
]))
areaCount++
if (isCleared) areaCleared++
if (pageAreas[i].task !== '') {
tile.classList.add('has-task')
}
if (pageAreas[i].boost > 1) {
tile.classList.add('has-boost')
}
}
mapPage.appendChild(_('text', ' '))
}
if (areaCleared === areaCount) allClearedCheckbox.checked = true
return mapPage
}
function getAreaTile(area, isCleared, areaLengthText, clearedCount) {
let areaText = area.skill
if (areaText === '-' && area.required > clearedCount) {
areaText = `${clearedCount}/${area.required}`
}
const tile = _('div', { className: 'maparea' }, [
isCleared ? _('div', { className: 'map_clear' }, [_('img', { src: '/mobile/images/progress_RewardIconClear.png' })]) : _('text', ''),
_('div', { className: 'map_icon' }, [
getRewardIcon(area.rewards.slice(-1)[0])
]),
_('div', { className: 'map_remain' }, [_('div', { className: 'map_remain_text' }, [_('text', areaLengthText)])]),
_('div', { className: 'map_skillseed_block font_0' }, [_('div', { className: 'map_skillseed_text' }, [_('span', {}, [_('text', areaText)])])]),
])
if (area.required > clearedCount) {
tile.appendChild(_('img', { className: 'map_lock_float', src: `${imageBase}/progress_LockCover.png` }))
} else if (area.hidden !== 0 && !isCleared) {
tile.appendChild(_('img', { className: 'map_lock_float', src: MAP_AREA_HIDDEN_LOCK }))
}
return tile
}
function getRewardIcon(reward) {
switch (reward.type) {
case 0: // gamePoint
case 4: // skillSeed
return _('div', { className: 'map_icon_limitbreak' }, [_('text', reward.reward)])
case 2: // trophy
{
const [bgImg, trophyName] = getTrophyInfo(reward.reward)
return _('div', { className: 'map_icon_trophy', style: { backgroundImage: `url(${bgImg})` } }, [
_('div', { className: 'map_text_trophy text_b' }, [_('div', { className: 'map_honor_text' }, [_('span', {}, [_('text', trophyName)])])])
])
}
case 1: // ticket
{
if (reward.id >= 20008000) {
return _('div', { className: 'map_icon_limitbreak' }, [getRewardIconImg(reward.id, reward.reward)])
}
}
case 5: // namePlate
case 8: // systemVoice
{
return _('div', { className: 'map_icon_nameplate' }, [getRewardIconImg(reward.id, reward.reward)])
}
case 3: // chara
{
return _('div', { className: 'map_icon_chara' }, [getRewardIconImg(reward.id, reward.reward)])
}
case 6: // music
case 12: // ultima
{
return _('div', { className: 'map_icon_music' }, [getRewardIconImg(reward.id, reward.reward)])
}
case 7: // mapIcon
case 9: // avatarAccessory
{
return _('div', { className: 'map_icon_avatar' }, [getRewardIconImg(reward.id, reward.reward)])
}
}
return _('div', { className: 'map_icon_limitbreak' }, [_('text', reward.reward)])
}
function getRewardIconImg(id, text) {
return _('img', { src: `https://estertion.win/__private__/chuni-rewards-${server}/${id}.webp`, event: { error: e => {
e.target.parentNode.replaceChild(_('text', text), e.target)
} }})
}
function getTrophyInfo(text) {
const sepIdx = text.indexOf('|')
if (sepIdx === -1) return [`${imageBase}/Title_normal.png`, text]
const trophyType = text.slice(0, sepIdx)
let trophyName = text.slice(sepIdx + 1)
let trophyTypeStr = 'normal';
switch (trophyType) {
case '0': trophyTypeStr = 'normal'; break
case '1': trophyTypeStr = 'copper'; break
case '2': trophyTypeStr = 'silver'; break
case '3': trophyTypeStr = 'gold'; break
case '4': trophyTypeStr = 'gold'; break
case '5': trophyTypeStr = 'platina'; break
case '6': trophyTypeStr = 'platina'; break
case '7': trophyTypeStr = 'rainbow'; break
// case '8': trophyTypeStr = 'rainbow'; break
case '9': trophyTypeStr = 'staff'; break
case '10': trophyTypeStr = 'ongeki'; break
case '11': trophyTypeStr = 'maimai'; break
default: trophyName = text;
}
return [`${imageBase}/Title_${trophyTypeStr}.png`, trophyName]
}
const initialPages = {}
function getPage(mapTitle, page) {
if (initialPages[mapTitle] && initialPages[mapTitle][page]) {
return mapBlockMap[mapTitle] = initialPages[mapTitle][page]
}
const map = mapInfo.find(m => m.name === mapTitle)
if (!map) {
alert(`未找到地图\n${mapTitle}\n第 ${page} 页`)
throw new Error(`未找到地图\n${mapTitle}\n第 ${page} 页`)
}
return mapBlockMap[mapTitle] = generatePage(map, page)
}
function addClickHandler() {
const mapBlocks = [...document.querySelectorAll('.map_block')]
mapBlocks.forEach(block => {
const mapTitle = block.querySelector('.map_title_text').textContent.trim()
const isInfinite = mapTitle.indexOf('ep. ∞') !== -1
mapBlockMap[mapTitle] = block
const map = mapInfo.find(m => m.name === mapTitle)
const currentPage = block.querySelector('.map_title_page_num').textContent *1
initialPages[mapTitle] = initialPages[mapTitle] || { [currentPage]: block, currentPage }
const totalPage = isInfinite ? Infinity : map ? map.areas.slice(-1)[0].page+1 : block.querySelector('.map_title_page_den').textContent *1
block.insertBefore(_('div', { className: ['map-paginator', currentPage>1?'has-prev':'', currentPage<totalPage?'has-next':''].join(' ') }, [
_('span', { className: 'prev', event: { click: _ => { block.parentNode.replaceChild(getPage(mapTitle, currentPage - 1), block); executeMarquee() } } }, [_('text', '<')]),
_('span', { className: 'next', event: { click: _ => { block.parentNode.replaceChild(getPage(mapTitle, currentPage + 1), block); executeMarquee() } } }, [_('text', '>')]),
]), block.firstChild)
const tiles = [...block.querySelectorAll('.maparea_block')]
const cleared = block.getElementsByClassName('map_clear')
storeMapPageProgress(mapTitle, currentPage, true)
tiles.forEach((tile, i) => {
const area = map?.areas.find(a => a.page === currentPage-1 && a.position === i)
if (!tile.querySelector('.maparea')) {
if (!area) return
while (tile.firstChild) tile.removeChild(tile.firstChild)
tile.appendChild(getAreaTile(area, false, area.rewards.slice(-1)[0].position, cleared.length))
}
if (area) {
if (area.task !== '') {
tile.classList.add('has-task')
}
if (area.boost > 1) {
tile.classList.add('has-boost')
}
}
const remainingTiles = tile.querySelector('.map_remain_text').textContent *1
tile.classList.add('user_data_friend_tap')
tile.addEventListener('click', () => showMapAreaDetail(mapTitle, currentPage, i, cleared.length, remainingTiles))
storeMapTileProgress(mapTitle, currentPage, i, remainingTiles, tile.querySelector('.map_clear') !== null)
})
})
}
const containerProgressing = document.querySelector('.box01').parentNode.appendChild(_('div', { className: 'box01 w420' }, [
_('div', { className: 'box01_title text_b' }, [_('text', '其他地图 - 进行中')])
]))
const containerNotStarted = document.querySelector('.box01').parentNode.appendChild(_('div', { className: 'box01 w420' }, [
_('div', { className: 'box01_title text_b' }, [_('text', '其他地图 - 未开始')])
]))
const containerFinished = document.querySelector('.box01').parentNode.appendChild(_('div', { className: 'box01 w420' }, [
_('div', { className: 'box01_title text_b' }, [_('text', '其他地图 - 已完成')])
]))
function addRemainingMaps() {
[containerProgressing, containerNotStarted, containerFinished].forEach(c => {
while (c.children.length > 1) {
const child = c.children[1]
const mapTitle = child.querySelector('.map_title_text').textContent
delete initialPages[mapTitle]
child.remove()
}
});
mapInfo.forEach(map => {
const mapTitle = map.name
if (initialPages[mapTitle]) return
let showingPage = map.areas.slice(-1)[0].page + 1
let finishedAreaCount = 0
let notStartedCount = 0
map.areas.forEach(area => {
const progress = getMapTileProgress(mapTitle, area.page + 1, area.position)
if (progress.position === 0 && !progress.finished) notStartedCount++
if (!progress.finished) showingPage = Math.min(showingPage, area.page + 1)
else finishedAreaCount++
})
initialPages[mapTitle] = { currentPage: showingPage }
if (finishedAreaCount === map.areas.length) {
mapBlockMap[mapTitle] = containerFinished.appendChild(generatePage(map, showingPage))
} else if (notStartedCount === map.areas.length) {
mapBlockMap[mapTitle] = containerNotStarted.appendChild(generatePage(map, showingPage))
} else {
mapBlockMap[mapTitle] = containerProgressing.appendChild(generatePage(map, showingPage))
}
})
containerProgressing.style.display = containerProgressing.children.length < 2 ? 'none' : ''
containerNotStarted.style.display = containerNotStarted.children.length < 2 ? 'none' : ''
containerFinished.style.display = containerFinished.children.length < 2 ? 'none' : ''
}
function addGlobalStyle() {
document.head.appendChild(_('style', {}, [_('text', [
'html.showing-detail{overflow:hidden}',
'.map-detail-background{',
' position:fixed;',
' top:0;',
' left:0;',
' width:100vw;',
' height:100vh;',
' background-color:rgba(0,0,0,0.5);',
'}',
'.map-detail{',
' margin: 0 auto;',
' background-color: white;',
' padding: 30px 20px 40px;',
' cursor: default;',
'}',
'.map-detail .map-reward.reward-got{',
' color: #AAA;',
'}',
'.bonus-target-cell{',
' padding: 0 0.5em;',
' max-width: 320px;',
' word-break: break-all;',
'}',
'.reward-name-cell{',
' max-width: 250px;',
' word-break: break-all;',
'}',
'.map-paginator{ position:relative; user-select:none; -webkit-user-select:none; }',
'.map-paginator span{',
' font-weight: bold;',
' position: absolute;',
' color: white;',
' margin-top: 4px;',
' text-shadow: 1px 0px 2px black,-1px 0px 2px black,0px 1px 2px black,0px -1px 2px black;',
'}',
'.map-paginator span.prev{ right: 3.5em; }',
'.map-paginator span.next{ right: -0.2em; }',
'.map-paginator:not(.has-prev) span.prev{ display:none; }',
'.map-paginator:not(.has-next) span.next{ display:none; }',
'.maparea_block.has-task .maparea::after{',
' content:"";position:absolute;top:4px;left:0px;width:78px;height:26px;',
' background-image:url('+MAP_AREA_TASK+');',
'}',
'.maparea_block.has-boost .map_icon::after{',
' content:"";position:absolute;top:-2px;right:0;width:60px;height:56px;background-size:100%;',
' background-image:url('+MAP_AREA_BOOST+');',
'}',
'.map_lock_float {',
' position:absolute;top:2px;left:3px;pointer-events:none;opacity:0.5;z-index:1;',
'}',
'.map_title_page_infinit img { vertical-align: middle }',
'.hidden_type_desc { color:#555;font-size:0.7em; }',
].join('\n'))]))
}
function executeMarquee() {
document.head.appendChild(_('script', {}, [_('text', `
$(function() {
// 許容する最大のWidth
var widthMax = 104;
$('.map_skillseed_text,.map_honor_text').each(function() {
var span = $(this).children('span');
// 称号文字列を表示しているSpanのWidth取得
var textWidth = span.innerWidth();
// Widthが許容値を超えていればマーキーさせる
if (textWidth > widthMax) {
$(this).marquee({speed : 60});
}
});
});
`)])).remove()
}
function showMapAreaDetail(mapTitle, currentPage, index, clearedCount, remainingTiles) {
const map = mapInfo.find(m => m.name === mapTitle)
if (!map) return alert('未找到地图')
const isInfinite = mapTitle.indexOf('ep. ∞') !== -1
const areasInPage = map.areas.filter(a => a.page === currentPage-1)
const area = map.areas.find(a => a.page === ((isInfinite && currentPage > 3) ? 3 : currentPage)-1 && a.position === index)
if (!area) return alert('未找到区域')
const areaIndexInPage = areasInPage.indexOf(area)
const hasNextArea = areaIndexInPage < areasInPage.length - 1
const hasPrevArea = areaIndexInPage > 0
const progress = getMapTileProgress(mapTitle, currentPage, index)
const totalPage = map.areas.slice(-1)[0].page + 1
const shrinkLength = area.shrink.slice(0, clearedCount).reduce((a,b)=>a+b, 0)
const length = area.rewards.slice(-1)[0].position - shrinkLength
const position = progress.finished ? length : Math.max(length + (PAGE_DATA_IS_SHRINKED ? 0 : shrinkLength) - remainingTiles, 0)
document.body.parentNode.classList.add('showing-detail')
let finishedCheckbox
const detailContainer = document.body.appendChild(_('div', { className: 'map-detail-background', event: { click: _ => { detailContainer.remove(); document.body.parentNode.classList.remove('showing-detail') } } }, [
_('div', { className: 'map-detail w460 text_l' }, [
_('div', { className: 'text_b' }, [_('text', map.name)]),
_('div', {}, [_('text', `第 ${currentPage}/${totalPage} 页,第 ${index+1} 格`)]),
_('div', {}, [_('label', { event: { click: e => e.stopPropagation() } }, [
finishedCheckbox = _('input', { type: 'checkbox', event: { change: __ => {
storeMapTileProgress(mapTitle, currentPage, index, null, finishedCheckbox.checked)
storeMapPageProgress(mapTitle, currentPage)
const tile = mapBlockMap[mapTitle].querySelectorAll('.maparea_block')[index]
if (finishedCheckbox.checked) {
tile.querySelector('.maparea').appendChild(_('div', { className: 'map_clear' }, [_('img', { src: '/mobile/images/progress_RewardIconClear.png' })]))
} else {
const clear = tile.querySelector('.map_clear')
if (clear) clear.remove()
}
detailContainer.remove()
tile.click()
} } }),
_('text', '区域已完成'),
getHiddenType(area.hidden, '区域'),
])]),
_('br'),
(hasPrevArea || hasNextArea) ? _('div', {}, [
_('div', { style: { display: 'flex', flexDirection: 'row', textDecoration: 'underline', color: '#32AAB4' }}, [
hasPrevArea ? _('span', { event: { click: async __ => { await sleep(); mapBlockMap[mapTitle].querySelectorAll('.maparea_block')[areasInPage[areaIndexInPage -1].position].click() } }, style: { flex: 1, padding: '5px 1em', cursor: 'pointer' }}, [_('text', '前一个区域')]) : new Comment('prev page'),
hasNextArea ? _('span', { event: { click: async __ => { await sleep(); mapBlockMap[mapTitle].querySelectorAll('.maparea_block')[areasInPage[areaIndexInPage +1].position].click() } }, style: { flex: 1, padding: '5px 1em', cursor: 'pointer', textAlign: 'right' }}, [_('text', '后一个区域')]) : new Comment('next page'),
]),
_('br'),
]) : new Comment('sibling page nodes'),
_('div', {}, [_('text', area.required > 0 ? `区域解锁:${clearedCount} / ${area.required}` : '')]),
_('div', {}, [_('text', area.boost > 1 ? `跑图加成x${area.boost}` : '')]),
_('div', {}, area.shrink.map((v,i)=>[i,v]).filter(v => v[1]>0).map(([count, shrink]) => _('div', {}, [_('text', `完成${count+1}个区域后缩短${shrink}格`)]))),
_('table', {}, area.bonus.map(showBonus)),
_('br'),
_('div', {}, [_('text', `当前位置:${position} / ${length}`)]),
_('table', {}, [_('tbody', {}, area.rewards.map(reward => {
const posNodes = []
const remainNodes = []
let rewardGot = false
if (reward.position >= length && shrinkLength > 0) {
if (reward.position - shrinkLength === length) {
// 最终奖励
posNodes.push(_('del', {}, [_('text', reward.position + '格')]))
posNodes.push(_('br'))
posNodes.push(_('text', length + '格'))
remainNodes.push(_('del', {}, [_('text', '还剩'+(reward.position - position)+'格')]))
remainNodes.push(_('br'))
remainNodes.push(_('text', progress.finished ? '已取得' : ('还剩'+(length - position)+'格')))
rewardGot = progress.finished
} else {
// 其他奖励倍缩短后删除
posNodes.push(_('del', {}, [_('text', reward.position + '格')]))
remainNodes.push(_('del', {}, [_('text', '还剩'+(reward.position - position)+'格')]))
rewardGot = true
}
} else {
rewardGot = reward.position <= position
posNodes.push(_('text', reward.position + '格'))
remainNodes.push(_('text', rewardGot ? '已取得' : ('还剩'+(reward.position - position)+'格')))
}
return _('tr', { className: 'map-reward '+(rewardGot ? 'reward-got' : 'reward-not-got') }, [
_('td', { className: 'text_r' }, posNodes),
_('td', { className: 'text_r', style:{padding:'0 0.5em'} }, remainNodes),
_('td', { className: 'text_l reward-name-cell' }, [_('text', reward.reward)]),
])
}))]),
_('div', {}, [_('text', area.task ? `课题曲:${area.task}` : '')]),
])
]))
finishedCheckbox.checked = progress.finished
}
function showBonus(bonus) {
switch (bonus.type) {
case 'chara':
return _('tr', {}, [
_('td', { className: 'text_r'}, [_('text', '使用角色')]),
_('td', { className: 'bonus-target-cell' }, [_('text', bonus.target)]),
_('td', {}, [_('text', `+${bonus.point}`)]),
])
case 'charaWorks':
return _('tr', {}, [
_('td', { className: 'text_r'}, [_('text', '使用分类角色')]),
_('td', { className: 'bonus-target-cell' }, [_('text', bonus.target)]),
_('td', {}, [_('text', `+${bonus.point}`)]),
])
case 'skill':
return _('tr', {}, [
_('td', { className: 'text_r'}, [_('text', '使用技能')]),
_('td', { className: 'bonus-target-cell' }, [_('text', bonus.target)]),
_('td', {}, [_('text', `+${bonus.point}`)]),
])
case 'skillCategory':
return _('tr', {}, [
_('td', { className: 'text_r'}, [_('text', '使用技能类型')]),
_('td', { className: 'bonus-target-cell' }, [_('text', bonus.target)]),
_('td', {}, [_('text', `+${bonus.point}`)]),
])
case 'music':
return _('tr', {}, [
_('td', { className: 'text_r'}, [_('text', '游玩歌曲')]),
_('td', { className: 'bonus-target-cell' }, [_('text', bonus.target)]),
_('td', {}, [_('text', `+${bonus.point}`)]),
])
case 'musicGenre':
return _('tr', {}, [
_('td', { className: 'text_r'}, [_('text', '游玩分类歌曲')]),
_('td', { className: 'bonus-target-cell' }, [_('text', bonus.target)]),
_('td', {}, [_('text', `+${bonus.point}`)]),
])
case 'musicWorks':
return _('tr', {}, [
_('td', { className: 'text_r'}, [_('text', '游玩来源歌曲')]),
_('td', { className: 'bonus-target-cell' }, [_('text', bonus.target)]),
_('td', {}, [_('text', `+${bonus.point}`)]),
])
case 'musicDif':
return _('tr', {}, [
_('td', { className: 'text_r'}, [_('text', '游玩难度')]),
_('td', { className: 'bonus-target-cell' }, [_('text', bonus.target)]),
_('td', {}, [_('text', `+${bonus.point}`)]),
])
case 'musicLv':
return _('tr', {}, [
_('td', { className: 'text_r'}, [_('text', '游玩歌曲等级')]),
_('td', { className: 'bonus-target-cell' }, [_('text', bonus.target)]),
_('td', {}, [_('text', `+${bonus.point}`)]),
])
case 'charaRank':
return _('tr', {}, [
_('td', { className: 'text_r'}, [_('text', '角色等级大于')]),
_('td', { className: 'bonus-target-cell' }, [_('text', bonus.target)]),
_('td', {}, [_('text', `+${bonus.point}`)]),
])
}
return _('div', {}, [_('text', `${bonus.type} ${bonus.target} +${bonus.point}`)])
}
function storeMapPageProgress(title, page, isInitialData = false) {
const map = mapInfo.find(m => m.name === title)
if (!map) return
page--
map.areas.forEach(area => {
if (area.page < page) {
storeMapTileProgress(title, area.page + 1, area.position, isInitialData ? area.rewards.slice(-1)[0].position : null, true)
} else if (area.page > page) {
storeMapTileProgress(title, area.page + 1, area.position, isInitialData ? 0 : null, false)
}
})
if (title.indexOf('ep. ∞') !== -1 && page > 3) {
const infiniteTile = map.areas.slice(-1)[0]
for (let i=infiniteTile.page; i<page; i++) {
storeMapTileProgress(title, i + 1, infiniteTile.position, isInitialData ? infiniteTile.rewards.slice(-1)[0].position : null, true)
}
}
}
function storeMapTileProgress(title, page, index, position, finished) {
const data = getStoredMapData()
data[title] = data[title] || {}
if (position === null) {
position = data[title][`${page}-${index}`] && data[title][`${page}-${index}`][0] || 0
}
data[title][`${page}-${index}`] = [position, finished]
storeMapData(data)
}
function getMapTileProgress(title, page, index) {
const data = getStoredMapData()
const [position, finished] = data[title] && data[title][`${page}-${index}`] || [0, false]
return { position, finished }
}
const localStorageProgressDataKey = 'CNMD_maps_progress'
function getStoredMapData() {
const data = localStorage[localStorageProgressDataKey]
return data ? JSON.parse(data) : {}
}
function storeMapData(data) {
localStorage[localStorageProgressDataKey] = JSON.stringify(data)
}
loadLocalInfo()
checkUpdateForLocalInfo()
addClickHandler()
addRemainingMaps()
addGlobalStyle()