chuni-net - Map Details

Display map details on chunithm-net

您需要先安裝使用者腳本管理器擴展,如 TampermonkeyGreasemonkeyViolentmonkey 之後才能安裝該腳本。

You will need to install an extension such as Tampermonkey to install this script.

您需要先安裝使用者腳本管理器擴充功能,如 TampermonkeyViolentmonkey 後才能安裝該腳本。

您需要先安裝使用者腳本管理器擴充功能,如 TampermonkeyUserscripts 後才能安裝該腳本。

你需要先安裝一款使用者腳本管理器擴展,比如 Tampermonkey,才能安裝此腳本

您需要先安裝使用者腳本管理器擴充功能後才能安裝該腳本。

(我已經安裝了使用者腳本管理器,讓我安裝!)

你需要先安裝一款使用者樣式管理器擴展,比如 Stylus,才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展,比如 Stylus,才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展,比如 Stylus,才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展後才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展後才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展後才能安裝此樣式

(我已經安裝了使用者樣式管理器,讓我安裝!)

// ==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.7
// @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
  let showSkillIcon = area.skill !== '-'
  if (areaText === '-' && area.required > clearedCount) {
    areaText = `${clearedCount}/${area.required}`
    showSkillIcon = false
  }
  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' }, [
      !showSkillIcon ? new Comment('skill icon') : _('div', { className: 'map_skillseed_img' }, [
        _('img', { src: `/mobile/images/SkillSeed_${area.skillIcon}.png` }),
        _('img', { src: `/mobile/images/SkillSeed_VersionPop_v${area.skillVer}.png` }),
      ]),
      _('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)])
      }
    case 13: // stage
      {
        return _('div', { className: 'map_icon_stage' }, [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);',
    '}',
    '@supports (backdrop-filter: none) {',
    '  .map-detail-background{',
    '    background: none;',
    '    backdrop-filter: brightness(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()