因BWiki近期更新,普通用户无法直接访问除重定向、编辑表格外的特殊页面(如最近更改等),进行3次有效编辑后即可恢复。
全站通知:

地图工具

阅读

    

2021-11-28更新

    

最新编辑:瓦瑞克Warrick

阅读:

  

更新日期:2021-11-28

  

最新编辑:瓦瑞克Warrick

来自战争雷霆WIKI_BWIKI_哔哩哔哩
跳到导航 跳到搜索
此模板为一键安装导入,不建议手动修改。

简单说明

说明:大地图工具,用于开放世界类型的游戏

说明:本工具属于大型工具,本页面不对源码进行展开,需要二改的可以前往原址自行研究

使用说明

第一步:搬运框架

将如下两个文件导入您的WIKI(特殊:导入页面

Map3.0Widget.xml

包含如下页面:

<style> .animate-fast{ transition: all 300ms; } #map-wrap #map-menu.toggle-active{z-index:99999} #map-wrap #map-menu .wiki-logo{ padding: 6px 0; width: 60%; } #map-wrap #map-menu .wiki-logo img{ width: 100%; } #map-wrap #map-menu .menu-search-box { padding: 14px 0px 0 0; width: 40%; } #map-wrap #map-menu .menu-search-box input { padding: 7px 12px; width: 100%; border: 1px solid rgba(0,0,0,0.2); border-radius: 20px; text-align: right; outline: none; margin-right: -4px; } #map-wrap #map-menu .menu-search-box .map-icon.search{ filter: brightness(0.5); } #map-wrap #map-menu .menu-btn-group{ display: flex; text-align: center; color: rgba(0,0,0,0.85); font-size: 14px; border: 1px solid rgba(0,0,0,0.1); line-height: 18px; padding: 9px 0; border-radius: 6px; width: 100%; } #map-wrap #map-menu .menu-btn-group> div{ cursor: pointer; width: 25%; border-right: 1px solid rgba(0,0,0,0.1); } #map-wrap #map-menu .menu-btn-group> div:hover{ color: rgba(0,0,0,1); } #map-wrap #map-menu .menu-btn-group> div:last-child{ border-right: none; } #map-wrap #map-menu .menu-tab-nav{ display: flex; text-align: center; font-size: 16px; font-weight: bold; color: rgba(0,0,0,0.35); width: 100%; justify-content: space-around; } #map-wrap #map-menu .menu-tab-nav> div{ cursor: pointer; width: 30%; padding: 10px; } #map-wrap #map-menu .menu-tab-nav> div:hover{ color: rgba(0,0,0,0.55); } #map-wrap #map-menu .menu-tab-nav > div.toggle-active{ color: rgba(0,0,0,0.85); position: relative; } #map-wrap #map-menu .menu-tab-nav:after{ content: ""; position: absolute; display: inline-block; width: 20px; border-bottom: 2px solid rgba(0,0,0,0.85); height: 0; bottom: -1px; left: 20%; margin-left: -10px; transition: all 300ms; } #map-wrap #map-menu .menu-tab-nav.tab-1.tab-index-1:after{ left: 50%; } #map-wrap #map-menu .menu-tab-nav.tab-2.tab-index-1:after{ left: 27%; } #map-wrap #map-menu .menu-tab-nav.tab-2.tab-index-2:after{ left: 72%; } #map-wrap #map-menu .menu-tab-nav.tab-3.tab-index-1:after{ left: 20%; } #map-wrap #map-menu .menu-tab-nav.tab-3.tab-index-2:after{ left: 50%; } #map-wrap #map-menu .menu-tab-nav.tab-3.tab-index-3:after{ left: 80%; } #map-wrap #map-menu .menu-tab-nav.tab-4.tab-index-1:after{ left: 12.5%; } #map-wrap #map-menu .menu-tab-nav.tab-4.tab-index-2:after{ left: 37.5%; } #map-wrap #map-menu .menu-tab-nav.tab-4.tab-index-3:after{ left: 62.5%; } #map-wrap #map-menu .menu-tab-nav.tab-4.tab-index-4:after{ left: 87.5%; } #map-wrap #mapSetting{ box-shadow: 0 1px 5px rgba(0, 0, 0, 0.65); border-radius: 6px; } #map-wrap .map-setting-btn{ zoom: 0.8; width: 24px; height: 24px; color: #CED3DE; line-height: 24px; background: rgba(0, 0, 0, 0.8); padding: 4px; white-space: nowrap; cursor: pointer; } #map-wrap .map-setting-btn:first-child{ border-radius: 6px 6px 0 0; } #map-wrap .map-setting-btn:last-child{ border-radius: 0 0 6px 6px; } #map-wrap .map-setting-btn.toggle-active{ color: white; background: #98c138; } #map-wrap #map { background-color: #fff !important } #map-wrap .mark-wrap { max-width: none } #map-wrap .BMap_bubble_title { word-wrap: break-word; word-break: keep-all; white-space: normal !important } #map-wrap .BMap_bubble_content .map-header { line-height: 20px } #map-wrap .BMap_bubble_content .map-header .map-info-id { display: inline-block; line-height: 36px } #map-wrap .BMap_bubble_content .map-header .map-info-mark { background: #000; font-size: 14px; color: #fff; text-align: center; padding: 6px 14px; float: right } #map-wrap .BMap_bubble_content .map-header, #map-wrap .BMap_bubble_content .map-info-des, #map-wrap .BMap_bubble_content .map-info-image-wrap { margin: 10px 0 } #map-wrap .BMap_bubble_content .map-info-image-wrap img { width: 100% } #map-wrap .BMap_bubble_content .map-info-footer { margin-top: 30px } #map-wrap .BMap_bubble_content .map-info-footer .map-info-wikilink { display: inline-block; border: 1px solid #00b6ff; font-size: 14px; color: #00b6ff; text-align: center; padding: 8px 44px } #map-wrap .BMap_bubble_content .map-info-footer .map-info-videolink { background: #000; font-size: 14px; color: #fff; text-align: center; padding: 8px 44px; float: right } #map-wrap .anchorBL, #map-wrap .BMap_cpyCtrl { display: none } #map-wrap .toggle-type { display: none; position: absolute; right: 100px; top: 20px } #map-wrap .toggle-json { display: none; position: absolute; right: 200px; top: 20px } #map-wrap .toggle-id { display: none; position: absolute; right: 300px; top: 20px } #map-wrap .BMap_pop { margin-top: -40px } #map-wrap .map-link, #map-wrap .map-mark { float: right; box-sizing: border-box; display: block; border-radius: 5px; margin-right: 10px; text-align: center; line-height: 30px; font-size: 14px; font-weight: 700; height: 30px; margin-top: 10px; padding: 0 10px; text-decoration: none; cursor: pointer; color: #fff } #map-wrap .map-mark { background-color: #242424 } #map-wrap .map-link { background-color: #0069f8 } #map-wrap #map-menu { width: 400px; max-width: calc(100vw - 66px); position: absolute; left: -400px; top: 0; box-shadow: 0 2px 8px rgba(0, 0, 0, .5); box-sizing: border-box; flex-direction: column; transition: transform .5s; transform: translate(0%, 0%); background: white; z-index: 9999; padding-top: 150px; height: 100%; } #map-wrap #map-menu .mapMenu { width: 100%; height: 150px; display: flex; justify-content: flex-start; flex-wrap: wrap; padding: 0 20px; border-bottom: 1px solid rgba(0,0,0,0.08); position: absolute; top: 0; align-content: space-between; justify-content: space-between; box-sizing: border-box; } #map-wrap #map-menu .mapShrink { width: 48px; height: 60px; position: absolute; right: -48px; top: 24px; cursor: pointer; border-radius: 0 6px 6px 0; } #map-wrap #map-menu .mapShrink .shrink-inner { width: 36px; height: 48px; border-radius: 0 6px 6px 0; color: rgba(206,211,222,1); background: rgba(0,0,0,0.8); text-align: center; } #map-wrap #map-menu .mapSearch { margin: 0 20px 10px; } #map-wrap #map-menu .mapSearch span { font-size: 18px; color: #333; font-weight: 700 } #map-wrap #map-menu .markerList { box-sizing: border-box; flex: 1; overflow-x: hidden; overflow-y: auto; margin: 0 auto; padding: 10px ; height: 100%; } #map-wrap *::-webkit-scrollbar { width: 6px; background: rgba(225,229,235,1); } #map-wrap *::-webkit-scrollbar-thumb { background: rgba(174,183,201,1); } #map-wrap #map-menu .markerList .orderItem { margin-bottom: 15px; padding-bottom: 10px; border-bottom: 1px solid rgba(0,0,0,0.08); } #map-wrap #map-menu .markerList .orderItem .order-h5 { height: 24px; line-height: 24px; color: #333; cursor: pointer; position: relative; margin: 0 0 10px; } #map-wrap #map-menu .markerList .orderItem .order-h5 span{ font-size: 16px; color: rgba(0,0,0,0.85); font-weight: bold; padding-left: 10px; } #map-wrap #map-menu .markerList .orderItem .order-h5 .pcRun { display: block; width: 78px; height: 24px; line-height: 24px; position: absolute; right: 25px; top: 0; font-size: 12px; text-align: center; color: rgba(152,162,171,1); } #map-wrap #map-menu .markerList .orderItem .items-wrap { display: flex; flex-wrap: wrap; font-size: 13px; } @media screen and (max-width: 768px) { #map-wrap #map-menu .markerList .orderItem .items-wrap > div { font-size: 12px } } #map-wrap #map-menu .markerList .orderItem .items-wrap > div:hover { background: rgba(242,244,246,0.5); } #map-wrap #map-menu .markerList .orderItem .items-wrap > .item-active { border-radius: 6px; background: rgba(242,244,246,1); color: rgba(0,0,0,0.85); } #map-wrap #map-menu .markerList .orderItem .items-wrap > div { display: flex; justify-content: space-between; padding: 10px; width: calc(50% - 4px); margin: 2px; align-items: baseline;flex-wrap: wrap; } #map-wrap #map-menu .markerList .orderItem .items-wrap .catTit .list-img { width: 20px; padding-right: 12px; margin-top: -3px; box-sizing: content-box; } #map-wrap #map-menu.toggle-active { transform: translate(100%, 0%); } #map-wrap .allHide { background: url(https://patchwiki.biligame.com/images/ys/f/f5/h3d1ij0241voanqcxcf682gbijlq7ko.png); } #map-wrap .allScreen { background: url(https://patchwiki.biligame.com/images/ys/2/28/qhi5h9t3yqclpmf72vgp708abm41b51.png) } #map-wrap .btn-clearStorage { background: url(https://patchwiki.biligame.com/images/ys/0/0a/guety3byfxw1rjnwtswsbtd4cu2ooa6.png) } #map-wrap .logo { width: 112px !important; height: 116px !important; margin: 0px !important; background: url(https://patchwiki.biligame.com/images/ys/9/90/d8dtw61n8v9xj69x7wj45jxjx6wmdue.png) no-repeat center; } @media screen and (max-width: 1200px) { #map-wrap .logo { display: none; } } .custom-select { position: relative; font-family: Arial; } .custom-select select { display: none; } .select-selected { background: url(https://patchwiki.biligame.com/images/ys/3/39/p9cztxdzn8dknr3ii4ohhf481acvi5z.png); } .select-items div, .select-selected { color: #445566; font-size: 16px; font-weight: bold; text-align: center; line-height: 32px; width: 108px; height: 32px; cursor: pointer; user-select: none; } .select-items { position: absolute; top: 100%; left: 0; right: 0; z-index: 99; border: #702e00 3px double; border-radius: 5px; } .select-items div { width: 102px; } .select-hide { display: none; } .select-items div:hover, .same-as-selected { background-color: #8b7449; color: #fff; } /* -- 地图全屏css -- */ /* full screen */ .fs { object-fit: contain; position: fixed !important; top: 0px !important; right: 0px !important; bottom: 0px !important; left: 0px !important; box-sizing: border-box !important; min-width: 0px !important; max-width: 100% !important; min-height: 0px !important; max-height: 100% !important; width: 100% !important; height: 100% !important; transform: none !important; margin: 0px !important; z-index: 9999999; } @media screen and (max-width: 539px) { #map-wrap #map-menu .markerList .orderItem .items-wrap > .item-active { background: rgba(237,237,237,1); } #map-wrap #map-menu { left: calc(-100vw + 65px); } #map-wrap #map-menu .menu-btn-group { font-size: 12px; } #map-wrap #map-menu .menu-tab-nav { font-size: 14px; } #map-wrap #map-menu .markerList .orderItem .order-h5{margin:4px} #map-wrap #map-menu .markerList .orderItem .order-h5 span{font-size:14px;} #map-wrap #map-menu .mapMenu{padding:0 10px;} #map-wrap #map-menu .markerList .orderItem{padding-bottom:2px} #map-wrap > .swipe-panel.toggle-active:before {content: "";height: 4px;width: 42px;background: #E1E1E1;position: absolute;top: 6px;left: 50%;margin-left: -21px;border-radius: 50px;} #map-wrap > .swipe-panel {padding: 18px 14px;box-shadow: 0 -2px 8px rgb(0 0 0 / 50%);box-sizing: border-box;padding-bottom: 120px;} .swipe-panel{ top: 100%; } .swipe-panel.toggle-active { top: 80%; } .leaflet-popup-tip-container{ padding-top: 66px; box-sizing: border-box; margin-left:-21px; } .leaflet-popup-content-wrapper, .leaflet-popup-close-button{ visibility: hidden; } .leaflet-popup.leaflet-zoom-animated { top: -26px; margin-left: 1px; } .leaflet-popup.leaflet-zoom-animated .leaflet-popup-tip-container { animation-name: likes; animation-direction: alternate; animation-timing-function: linear; animation-delay: 0s; animation-iteration-count: infinite; animation-duration: 1s; } @keyframes likes { 0%{ transform: scale(1); } 25%{ transform: scale(0.9); } 50%{ transform: scale(0.85); } 75%{ transform: scale(0.9); } 100%{ transform: scale(1); } } } @media screen and (max-width: 320px) { #map-wrap #map-menu .menu-btn-group { font-size: 10px; padding: 5px 0; } #map-wrap #map-menu .menu-btn-group .map-icon:first-child{ display:none!important; } #map-wrap #map-menu .menu-tab-nav { font-size: 12px; } #map-wrap #map-menu .markerList .orderItem .order-h5 span{font-size:13px;} } </style>
<style> .map-icon{ width: 1.4em; height: 1.4em; margin: -0.2em -0.2em 0 -0.4em; display: inline-block!important; background-size: contain; vertical-align: middle; transition: all 300ms; } .map-area-tab .shrink-inner { border-radius: 0!important; } .map-area-tab .shrink-inner:first-child{ border-radius: 0 6px 0 0!important; } .map-area-tab .shrink-inner:last-child { border-radius: 0 0 6px 0!important; } .map-area-tab .shrink-inner .map-icon{ background-image:url(https://patchwiki.biligame.com/images/sesbq/b/b0/0w44s0dc5l16jkyhcc96vqq0mkhmoir.png); background-size: auto 30.4px; width: 2.2em; height: 2.2em; margin-top: 7px; opacity: 0.6; } .map-area-tab .shrink-inner:hover .map-icon {opacity:1} .map-area-tab .shrink-inner.item-active .map-icon {opacity:1} .map-icon.world { background-position: -1.2px; } .map-icon.mengde { background-position: 209.8px; } .map-icon.liyue { background-position: 383.2px; } .map-icon.xueshan { background-position: 351.5px; } .map-icon.daoqi { background-position: 174px; } .map-icon.h-flip{transform: scaleX(-1);} .map-icon.v-flip{transform: scaleY(-1);} .shrink-inner:hover .menu-arrow{color:white} .map-icon.search{background-image: url();} .map-icon.menu-arrow{background-image: url();} .map-icon.lang{background-image: url();} .map-icon.lang-arrow{background-image: url();} .map-icon.windowed{background-image: url();} .map-icon.fullscreen{background-image: url();} .map-icon.visible{background-image: url();} .map-icon.hidden{background-image: url();} .map-icon.list-arrow{background-image: url();} .map-icon.refresh{background-image: url();} </style>
<script src="https://wiki.biligame.com/ys/MediaWiki:Knockout.js?action=raw&ctype=text/javascript"></script> <style> .simple-folder{cursor:pointer} .simple-folder[data-expand=false] + *{ height: 0; overflow: hidden; transition-property: height; transition-duration: 1s;} .orderItem.collapsed > .items-wrap{display:none!important} .mapSearch{display:none!important} </style> <script name="map.api.js"> var BwikiMapApi = function () { var origin = 'https://tools-wiki.biligame.com' var param = { game: null, site: null, dataPrefix: 'Data:Map', i18n: 'zh' } var sign = function (data) { if (typeof data !== "object") throw Error("Sign Failed!") var sign = md5(data.game + (data.markTypes || "") + (data.i18n || "") + data.ts + "9b83b0c44c89e26fc52aa50a202c2fd1"); return Object.assign({}, data, {sign: sign}) } this.set = function (data) { Object.assign(param, data) } this.getOrderData = function (i18n, callback) { if (!param.game) throw Error("Required parameter missing: game") if (typeof i18n === 'function') { callback = i18n i18n = null } $.ajax({ method: "post", url: origin + '/wiki/getOrderData', data: sign({ game: param.game, i18n: i18n || param.i18n, ts: (new Date).getTime() }), success: function (data) { if (typeof callback === "function") callback(data.data) } }) } this.getMapData = function (markTypes, callback) { if (!param.game) throw Error("Please set the query parameters first: game") if (!markTypes) throw Error("Required parameter missing: markTypes") $.ajax({ method: "post", url: origin + '/wiki/getMapData', data: sign({ game: param.game, markTypes: markTypes, ts: (new Date).getTime() }), success: function (data) { if (typeof callback === "function") callback(data.data) } }) } this.getPointData = function (list, callback) { if(typeof list === "function"){ callback = list; list = ["point.json"] } $.each(list, function(i, page){ $.ajax({ url: `/${param.site}/${param.dataPrefix}/${page}`, success: function (data) { data = $(data).find("#mapPointData").text().replace(/:Data:[\s\S]{0,30}\/json/g, ":[]"); data = (new Function('return ' + (data || "{}")))(); if (typeof callback === "function") callback(data) console.log(data) } }) }) } } </script> <script name="map.game.switcher.template" id="switcher-template" type="text/html"> <span data-bind="text: label" style="display:inline-block;font-size: 13px;height: 1em; line-height: 1em;"></span> <br> <label data-bind="attr: { style: 'display: inline-block;font-size: 20px;height: 1em;width: 2em;border-radius: 1em;cursor: pointer;vertical-align: bottom;background: ' + ($root[selected]() ?color:'#aaa')}"> <div class="animate-fast" data-bind="attr: { style: 'height: 1em;width: 1em;border-radius: 1em;background: #FFF;' + ($root[selected]() ?'transform: translate3d(100%, 0, 0)':'') } "></div> </label> </script> <script name="map.game.toolbar.template" id="map-menu-template" type="text/html"> <style data-bind="if: showMenu()"> .leaflet-left { left: 400px; } </style> <div id="map-menu" data-bind="css: {'toggle-active': showMenu()}"> <div class="mapShrink" data-bind="click: function(){ showMenu(!showMenu()) }"> <div class="shrink-inner"> <i class="map-icon menu-arrow" data-bind="style: { margin: '1em 0',transform : showMenu()? 'scaleX(-1)' : 'scaleX(1)'}"></i> </div> </div> <div class="mapShrink map-area-tab" style="height:auto;top: calc(10vh + 80px);" data-bind="foreach: areaData, visible: areaData().length != 0"> <div data-bind="class: 'shrink-inner ' + ($root.selectedArea() === inside? 'item-active':'') "> <i data-bind="class: 'map-icon ' + icon , click: function(){$root.changeArea($index())}"></i> </div> </div> <div class="mapMenu"> <div class="wiki-logo"> <img src="https://patchwiki.biligame.com/resources/assets/images/logo/logo_ys.png?t=2021052411&v=41" alt=""> </div> <div class="menu-search-box"> <input data-bind="value: keyword, valueUpdate: ['afterkeydown', 'input'], click: openToolbox" style="text-align: left"/> <i class="map-icon search" style="width: 2em; height: 2em; position: absolute; right: 24px; top: 20px;"></i> </div> <div class="menu-btn-group"> <div title="此功能暂未完成"> <i class="map-icon lang"></i> <span>语言</span> <i class="map-icon lang-arrow" style="position: relative; left: 0.5em;"></i> </div> <div data-bind="click: purge"> <i class="map-icon refresh"></i> <span>刷新</span> </div> <div data-bind="click: fullScreenSwitcher"> <i data-bind="class: 'map-icon ' + (isFullScreen()? 'windowed' : 'fullscreen')"></i> <span data-bind="text: isFullScreen()?'退出全屏':'网页全屏'"></span> </div> <div data-bind="click: removeAllMarkers"> <i class="map-icon hidden"></i> <span>全隐藏</span> </div> </div> <div data-bind="class: 'menu-tab-nav tab-index-' + menuTabIndex() + ' tab-' + menuTabList().length, foreach: menuTabList"> <div data-bind="class: cls, text: label, click: function(){$root.menuTab(id);if(id==='markerPanel') $root.updateCollections();}"></div> </div> </div> <div id="menuList" class="markerList" data-bind="visible: menuTab() == 'listPanel', foreach: categoryObject"> <div data-bind="class: 'orderItem' + ($data.visible?'':' hide') + (!$data.collapsed?' collapsed':'')"> <div class="order-h5" data-bind="click: $root.collapse"><span data-bind="text: name"></span><small data-bind="text: '('+$data.count+')'" style="color:#aaa"></small> <i data-bind="class: 'map-icon list-arrow' + (!collapsed? '' : ' v-flip')" style="float: right; margin: 4px 10px 0 0;"></i> </div> <div class="items-wrap" data-bind="foreach: data "> <div data-bind="class:'select-item ' + (selected?'item-active':'') + (($root.toolbox() || pointsNum) && !hide?'':' hide'), click: $root.setMarkers" class="select-item" style="cursor:pointer"> <span class="catTit"> <img class="list-img" data-bind="attr: {src: icon}" alt="icon"> <span data-bind="text: markTypeName"></span> </span> <span class="catNum" data-bind="text: pointsNum"></span> </div> </div> </div> </div> <div class="markerList" data-bind="attr: {id:isPhone()?'':'infoPanel'},visible: !isPhone() && (menuTab() == 'infoPanel')"> 点击点位可以查看该点位的详细信息或进行讨论 </div> <div class="markerList" id="markerPanel" data-bind="visible: menuTab() == 'markerPanel'" style="padding:10px;"> <button class="btn btn-primary" style="background: rgb(95,108,121);width:calc(50% - 3px)" data-bind="click: saveUserData">保存数据至wiki</button> <button class="btn btn-primary" style="background: rgb(95,108,121);width:calc(50% - 3px)" data-bind="click: loadUserData">从wiki同步数据</button> <div class="drop-down-wrap panel panel-default" style="margin:10px 2px 10px 0;width:auto;padding: 0;font-size:12px;"> <div class="wrap-title panel-heading" style="height:auto"><div class="title" style="line-height:inherit;color:#aaa; font-size:12px;">之前地图的标记数据没有正常显示怎么办?</div><div class="glyphicon icon-arrow-close"></div></div> <div class="wrap-content panel-body" style="display: none;"> 目前新版地图与旧版地图的数据是分离的,若原地图还能正常区分已标记点位,请点击下方按钮尝试进行修复。<br> <div style="text-align:center;"><button class="btn btn-primary btn-xs" style="margin: 4px;" data-bind="click: refreshUserData">修复标记数据</button></div> 若修复后还未解决问题,请 联系技术支持 或 加入Q群:759910798 进行反馈 </div> </div> <div data-bind="class: 'orderItem'"> <div class="order-h5" data-bind="click: $root.collapse"><span data-bind="text: '统计' + ($root.selectedArea() !== ''? '(' + $root.selectedArea() + '地区)':'')"></span></div> <div class="items-wrap" data-bind="foreach: collections "> <div data-bind="class:'select-item ' + (pointsNum?'':' hide'), click: $root.setMarkers" style="width: 100%;padding-left: 44px;font-size: 13px;cursor:pointer; position:relative"> <img class="list-img" data-bind="attr: {src: icon}" alt="icon" style="height: 34px;width: 24px;position: absolute;left: 10px;"> <span class="catTit" style="display: flex;justify-content: space-between;flex-wrap: wrap;width: 100%;"> <span data-bind="text: markTypeName"></span> <span class="catNum" data-bind="text: count + ' / ' + pointsNum"></span> </span> <div class="progress" style="width: 100%;margin: 4px 0 0 0;height: 10px;"> <div class="progress-bar" role="progressbar" aria-valuemin="0" data-bind="style: { width: count / pointsNum * 100 + '%', background: 'rgb(95,108,121)' } " > <span></span> </div> </div> </div> </div> </div> <div class="drop-down-wrap panel panel-default" style="margin:10px 2px 10px 0;width:auto;padding: 0;font-size:12px;"> <div class="wrap-title panel-heading" style="height:auto"><div class="title" style="line-height:inherit;color:#aaa; font-size:12px;">点位数据有误,如何编辑点位?</div><div class="glyphicon icon-arrow-close"></div></div> <div class="wrap-content panel-body" style="display: none;"> 新版地图的地图点位由玩家进行维护,欢迎加入Q群:759910798 讨论并获取打点教程 </div> </div> <div class="drop-down-wrap panel panel-default" style="margin:10px 2px 10px 0;width:auto;padding: 0;font-size:12px;"> <div class="wrap-title panel-heading" style="height:auto"><div class="title" style="line-height:inherit;color:#aaa; font-size:12px;">是否可以补充自定义点位?</div><div class="glyphicon icon-arrow-close"></div></div> <div class="wrap-content panel-body" style="display: none;"> 目前该功能已纳入开发考虑,但因人手不足,欢迎前端开发加入一起完善,可加入Q群:759910798 进行讨论 </div> </div> </div> <div id="searchPanel" data-bind="visible: menuTab() == 'searchPanel'"> 搜索结果 </div> <div id="toolPanel" class="toolPanel" data-bind="visible: menuTab() == 'toolPanel'"> <div data-bind="if: hasGeomanToolbar() && selectedType()"> <style>.leaflet-grab, .leaflet-interactive { cursor: default;}</style> 当前标记: <img data-bind="attr: {src: selectedTypeData().iconUrl}" alt="icon" style="height:18px"> <span data-bind="text: selectedTypeData().typeName"></span> <input id="saveSummary" placeholder="修改摘要" style="width:150px"/> <div style="background: rgba(0,0,0,0.85);color: white;border-radius: 6px;padding: 4px;display: inline-block;font-size: 12px; cursor: pointer" data-bind="click: savePoints">保存 </div> <div class="simple-folder" data-expand="true">操作历史</div> <div data-bind="foreach: markerOperationList" style="max-height: 50%;overflow: auto;"> <div data-bind="text: $data"></div> </div> <div class="simple-folder" data-expand="true">点位</div> <div id="mapDataGrid" data-bind="if: hasGeomanToolbar() && selectedType()" style="max-height:50%;overflow: auto;"> <table style="width: 100%; word-break: break-all;font-size: 8px"> <thead> <tr> <th data-bind="text: 'No.'"></th> <th data-bind="text: 'id'"></th> <th data-bind="text: 'uid'"></th> <th data-bind="text: 'lat, lng'"></th> </tr> </thead> <tbody data-bind="foreach: markersData()[selectedType()]"> <tr data-bind="style: $root.stateColor($data.state)"> <td data-bind="text: $index() + 1"></td> <td></td> <td data-bind="html: $data.uid || ''"></td> <td data-bind="html: point.lat + ', ' + point.lng"></td> </tr> </tbody> </table> </div> </div> <div data-bind="if: hasGeomanToolbar() && !selectedType()"> <span data-bind="text: selectedTypeData().typeName"></span> <input id="saveName" placeholder="路径名" style="width:100px"/> <input id="saveSummary" placeholder="修改摘要" style="width:150px"/> <div style="background: rgba(0,0,0,0.85);color: white;border-radius: 6px;padding: 4px;display: inline-block;font-size: 12px; cursor: pointer" data-bind="click: saveGeoJSON">保存 </div> <div data-bind="text: editingGeoJSON" style="max-height:50%;overflow: auto;"></div> </div> <div class="simple-folder" data-expand="true">区域</div> <div style="max-height:50%;overflow: auto;"> <table style="width: 100%; word-break: break-all;font-size: 8px"> <thead> <tr> <th data-bind="text: '区域名称'"></th> <th data-bind="text: '操作'"></th> </tr> </thead> <tbody data-bind="foreach: areaData()"> <tr> <td data-bind="text: name"></td> <td ><div style="background: rgba(0,0,0,0.85);color: white;border-radius: 6px;padding: 4px;display: inline-block;font-size: 12px; cursor: pointer" data-bind="click: function(){$root.loadGeoJSON($index())}">载入</div></td> </tr> </tbody> </table> </div> </div> </div> <div id="mapSetting" style="position: absolute; right: 10px ; bottom: 100px ;z-index: 1000;" data-bind="visible:toolbox"> <div class="map-setting-btn" data-bind="click:$root.positionPopupSwitcher, css: {'toggle-active': hasPositionPopup()}">坐标</div> <div class="map-setting-btn" data-bind="click:$root.getOldMapData, visible: hasGeomanToolbar()">GET</div> <div class="map-setting-btn" data-bind="click:$root.geomanToolbarSwitcher, css: {'toggle-active': hasGeomanToolbar()}">打点</div> <div class="map-setting-btn" data-bind="click:function(){$root.toolbox(false)}">关闭</div> </div> </script> <div id="map-div" name="map.body.html"> <div id="map-wrap" class="swipe-container" style="position:relative; overflow:hidden; height: calc(100vh - 50px);"> <div id="map3" style="height: 100%"></div><div class="swipe-panel"></div> <div class="swipe-panel" style="position:absolute; width:100%; height:100%; transition: top 0.5s; background: white;z-index: 99999;border-radius:20px" data-bind= "css: 'swipe-panel ' + ($root.selectedPoint() !== null? 'toggle-active':''),visible: isPhone()" ><div id="marker-info-panel"></div><div data-bind="attr: {id: isPhone() ? 'infoPanel':''}"></div></div> <div data-bind="template: { name: 'map-menu-template' }"></div> </div> </div> <script name="map.common.js"> var util = { autoSet: function (self, defaultData, data, mode) { /* ko初始化使用 */ data = Object.assign({}, defaultData, data); for (var i in data) self[i] = mode ? ko.observable(data[i]) : data[i] return self; }, loading: function () { $("body").append("<div class='loading-mask' style='position:fixed;top:0;left:0;width:100%;height:100%;background:rgba(0,0,0,0.1);z-index:99999999'></div>") }, loadend: function () { $(".loading-mask").remove(); }, getUrlValue: function (part, key) { var query = window.location[part].substring(1); var vars = query.split("&"); for (var i = 0; i < vars.length; i++) { var pair = vars[i].split("="); if (pair[0] === key) return pair[1] || ''; } return ''; }, getUID: function () { // 需查重 return Number(Math.random().toString().substr(3, 3) + Date.now()).toString(36) }, coordFormatter: function(latlng) { return {lat: latlng.lat.toFixed(6), lng: latlng.lng.toFixed(6)} } } </script> <script name="jquery.swipe"> var swipe = function () { var self = $(this), $children = $(this).children('.swipe-panel'), cNum = $children.length; var $target, index = 0, startY = 0, diffY = 0; self.on('touchstart', '.swipe-panel', function (event) { if (event.targetTouches.length > 1) return true; $target = $children.eq(index); startY = event.targetTouches[0].pageY; }); self.on('touchmove', '.swipe-panel', function (event) { // if ($children.index(event.target) !== 1) return false; if (event.targetTouches.length > 1) return true; diffY = startY - event.targetTouches[0].pageY; var top = 10; if ( diffY > 0) { top = (80 - (diffY / 6)); if (top > 80) top = 80; $target.next('.swipe-panel').css({ transition: "0s", top: top + "%", }) return false; } else if (index !== 0 && diffY < 0) { top = -(diffY / 6); if (top < 10) top = 10; $target.css({ transition: "0s", top: top + "%", }) return false; } else { return false; } }); self.on('touchend', '.swipe-panel', function (event) { // if ($children.index(event.target) !== 1) return false; if (event.targetTouches.length >= 1) return false; var minY = 20; var isSuccess = Math.abs(diffY) > minY; if (index < cNum - 1 && diffY > 0) { isSuccess ? index++ : 0; $target.next('.swipe-panel').css({ transition: "1s", top: isSuccess ? "10%" : "80%", }) return false; } if (index !== 0 && diffY < 0) { isSuccess ? index-- : 0; $target.css({ transition: "1s", top: isSuccess ? "80%" : "10%", }) return false; } }); }; </script> <script name="map.markerController.js"> var MapMarkerController = function () { } </script> <script name="map.editorController.js"> var MapEditorController = function () { } </script> <script name="map.model.js"> var MapModel = function (data) { var self = util.autoSet(this, this.defaultData = { mapCenter: [0, 0], zoom:5, maxZoom: 8, minZoom: 4, toolbox: false, hasPositionPopup: false, hasGeomanToolbar: false, categoryData: {}, categorySelected: {}, categoryObject: [], areaMarkersData : {}, selectedType: null, selectedPoint: null, selectPointObj: null, selectedTypeData: {}, keyword: "", menuTab: 'listPanel', markersData: {}, markerWithInfo: [], areaData: [], selectedArea: '', showMenu: true, isFullScreen: false, editingGeoJSON: "", dataList: ["point.json"], snappable:false, markedMarker: {} }, data, true); self.markerOperationList = ko.observableArray([]); self.collections = ko.observableArray([]); self.updateCollections = function(){ var result = self.categoryData(); self.collections([]) $.each(result, function(key, category){ var count = 0; $.each(category.data, function(tindex, type) { var collection = $.extend({count:0},type); $.each(self.markersData()[type.markType] || [], function(pindex, point) { if(type.collectible && checkMarked(type.markType, point.id) && (self.selectedArea() === "" || point.inside === self.selectedArea())) {collection.count++;} }) if(type.collectible) self.collections.push(collection)   }) }) } var setCategoryObject = function () { var result = self.categoryData(), markers = {}; $.each(result, function(key, category){ var count = 0, selected = 0; $.each(category.data, function(tindex, type) { var collection = $.extend({count:0},type); type.selected = self.categorySelected()[type.markType]; selected += type.selected?1:0 type.hide = false; if(type.markTypeName.indexOf(self.keyword()) === -1){ type.hide = true; return true; } markers[type.markType] = [] if(self.selectedArea() === "") { markers[type.markType] = self.markersData()[type.markType] || [] } else { $.each(self.markersData()[type.markType] || [], function(pindex, point) { if(point.inside === self.selectedArea()) markers[type.markType].push(point) }) } type.pointsNum = markers[type.markType].length; if( markers[type.markType].length > 0) count ++;   }) category.count = self.toolbox() ? category.data.length : count; category.visible = self.toolbox() || (count > 0);   if(category.collapsed === undefined || typeof category.collapsed === "number") {     category.collapsed = selected   } }) self.categoryObject([]); self.categoryObject(result); self.areaMarkersData([]); self.areaMarkersData(markers); } self.search = ko.computed(function () { setCategoryObject(); return self.keyword(); }, this); self.collapse = function (obj) { var result = self.categoryData(); $.each(result, function(key, category){ if(category.name === obj.name) category.collapsed = !category.collapsed }) self.categoryObject([]); self.categoryObject(result); } self.menuTabList = ko.computed(function () { var list = [{id: "listPanel", label: "点位列表", cls:self.menuTab()==="listPanel"?"toggle-active":""}]; if(!self.isPhone()) list.push({id: "infoPanel", label: "点位信息", cls:self.menuTab()==="infoPanel"?"toggle-active":""}) if(!self.toolbox()) list.push({id: "markerPanel", label: "我的标记", cls:self.menuTab()==="markerPanel"?"toggle-active":""}) // if(self.keyword() !== "") list.push({id: "searchPanel", label: "搜索结果", cls:self.menuTab()==="searchPanel"?"toggle-active":""}) if(self.toolbox()) list.push({id: "toolPanel", label: "控制台", cls:self.menuTab()==="toolPanel"?"toggle-active":""}) return list; }, this); self.menuTabIndex = ko.computed(function () { var index = 1; $.each(self.menuTabList(), function(i, tab) { if(tab.id === self.menuTab()) { index = i + 1; return false}; }); return index; }, this); var layers = {}, icons = {}, markerTypes = [], markerGroups = {}, markedGroups = {}, storageKey = ""; var posPopup = L.popup(); var reformatInfoPgae = function (html) { var $html = $(html), $div = $("<div>"); $html.find("a").attr("target", "_blank"); $html.appendTo($div); return '<style>.wiki-header, #content > *, #bodyContent > * , .comments-title, .comment-time {display:none} #wikiToapp{display:none!important} .mw-body{min-height:auto!important}body{background:white!important;}</style>' + $div.html() + '<style>#bodyContent, #mw-content-text, #flowthread{display:block} #flowthread > .comment-replybox {position: fixed;padding:0;bottom:0;background:white;z-index:99;width: calc(100% - 26px);} .comment-body{padding:0!important} .comment-thread .comment-replybox{margin:0}.comment-body textarea{height: 36px!important; line-height: 26px; padding: 6px 10px; outline: none; background: #F0F0F0; border-radius: 51px; margin-bottom: 4px; color: black;}.comment-body textarea::-webkit-scrollbar { display: none; }.comment-toolbar{height: 36px; margin-bottom: 4px; width: 60px; padding: 0; margin-left: 10px;}.comment-submit{height: 36px; line-height: 0; width: 60px; border-radius: 51px; background: rgba(0,0,0,0.85);}.comment-avatar{margin:-1px 4px 0 0}.comment-avatar img,.comment-thread>div:not(:first-child) .comment-avatar img{width:20px;height:20px}.comment-post{padding:8px 0 4px 0}.comment-footer{padding-top:4px}</style>' } self.changeArea = function(index) { var time1 = new Date().getTime(), time2 = 0; if(self.areaData()[index].inside && !self.areaData()[index].split) { splitIntoArea(index) } self.markerOperationList.push("分区 " + ((time2 = new Date().getTime()) - time1)) time1 = time2 var selectedTemp = mapModel.categorySelected(); self.selectedArea(self.areaData()[index].inside); self.markerOperationList.push("设置选择地区 " + ((time2 = new Date().getTime()) - time1)) time1 = time2 self.removeAllMarkers(); self.markerOperationList.push("移除全部标记 " + ((time2 = new Date().getTime()) - time1)) time1 = time2 setCategoryObject(); self.markerOperationList.push("重新计算点位列表 " + ((time2 = new Date().getTime()) - time1)) time1 = time2 $.each(selectedTemp, function(type, selected){ if(selected) self.setMarkers({markType: type}, true) }); self.markerOperationList.push("重新显示已选点位 " + ((time2 = new Date().getTime()) - time1)) time1 = time2 setCategoryObject(); self.markerOperationList.push("重新计算点位列表 " + ((time2 = new Date().getTime()) - time1)) time1 = time2 if(self.areaData()[index].center !== null) map.panTo(self.areaData()[index].center); else { map.panBy([1,1])} self.markerOperationList.push("移动中心 " + ((time2 = new Date().getTime()) - time1)) time1 = time2 self.updateCollections(); self.markerOperationList.push("重新计数 " + ((time2 = new Date().getTime()) - time1)) time1 = time2 } var splitIntoArea = function(index) { var area = self.areaData()[index]; var t1 = new Date().getTime(); $.each(mapModel.markersData(), function(type, testPoints){ $.each(testPoints, function(index, testPoint){ if(testPoint.inside) { return true; } var inside = isMarkerInsidePolygon(testPoint, area.geojson.features[0].geometry.coordinates[0]); if(inside) { testPoint.inside = area.name; return true;} }) }) area.split = true; //self.areaData(self.areaData().splice(index, 1, area)); self.areaData().splice(index, 1, area); var t2 = new Date().getTime(); console.log(area.name + " 分区所花时间 " + (t2 - t1)) } var isMarkerInsidePolygon = function(marker, poly) { var polyPoints = poly; var x = marker.point.lng, y = marker.point.lat; var inside = false; for (var i = 0, j = polyPoints.length - 1; i < polyPoints.length; j = i++) { var xi = polyPoints[i][0], yi = polyPoints[i][1]; var xj = polyPoints[j][0], yj = polyPoints[j][1]; var intersect = ((yi > y) != (yj > y)) && (x < (xj - xi) * (y - yi) / (yj - yi) + xi); if (intersect) inside = !inside; } return inside; } self.loadGeoJSON = function(index) { var layer = L.geoJSON(self.areaData()[index].geojson,{ style: function (feature) { return {color: feature.properties.color || "#3388ff"}; } }).addTo(map); layer.on('pm:markerdragend', markerEditController); } self.generateGeoJson = function(){ var fg = L.featureGroup(); var layers = findLayers(map); layers.forEach(function(layer){ fg.addLayer(layer); }); return JSON.stringify(fg.toGeoJSON()) } function findLayers(map) { var layers = []; map.eachLayer(layer => { if ( layer instanceof L.Polyline || layer instanceof L.Marker || layer instanceof L.Circle || layer instanceof L.CircleMarker ) { layers.push(layer); } }); // filter out layers that don't have the leaflet-geoman instance layers = layers.filter(layer => !!layer.pm); // filter out everything that's leaflet-geoman specific temporary stuff layers = layers.filter(layer => !layer._pmTempLayer); return layers; } var getPurgeTitles = function(){ var titles = [mw.config.get("wgPageName")]; $.each(self.dataList(), function(i, page) { titles.push(self.dataPrefix() + "/" + page) }) return titles.join("|") } self.purge = function(){ new mw.Api().post({ "action": "purge", "format": "json", "titles": getPurgeTitles() }).then(function(){ window.location.reload(); }) } self.getOldMapData = function () { var type = self.selectedType() if (!type) return false mapApi.getMapData(self.selectedType(), function (data) { alert("请到控制台(F12)查看 " + self.selectedTypeData().typeName + " 数据!"); $.each(data, function (i, p) { p.state = "fromDB" }) updateObject(self.markersData, type, {[type]: data}) console.log(data) }) } var clickCounter = 0, clickTimer = null; self.openToolbox = function () { if(clickTimer){clearTimeout(clickTimer)} clickTimer=setTimeout(function(){ clickCounter=0 },1000); if(++clickCounter>=3){ mapModel.toolbox(true) } } self.fullScreenSwitcher = function () { var $map = $("#map-wrap").toggleClass("fs"); $($map.parent()[0].tagName === "BODY" ? "#map-div" : "body").prepend($map); self.isFullScreen(!self.isFullScreen()) map.invalidateSize(false) } self.positionPopupSwitcher = function (disabled) { self.hasPositionPopup(disabled = !self.hasPositionPopup()) map[disabled ? 'on' : 'off']('click', function (e) { posPopup.setLatLng(e.latlng).setContent(e.latlng.toString()).openOn(map); }); if (!disabled) posPopup.remove() } self.geomanToolbarSwitcher = function (disabled) { self.hasGeomanToolbar(disabled = !self.hasGeomanToolbar()) map.pm[disabled ? 'addControls' : 'removeControls']({position: 'bottomleft', snappingOption: self.snappable()}); map.pm.setLang('zh'); disabled ? self.removeAllMarkers() : '' self.setMarkers = disabled ? setMarkersEditable : setMarkersCanvas } self.createPointPage = function () { var id = self.selectedPoint() var pointType = self.selectPointObj().icon.options.type var Text = "欢迎在下方留言~" var ChestText switch(pointType){ case 60:ChestText = "华丽的宝箱";break; case 61:ChestText = "精致的宝箱";break; case 62:ChestText = "普通的宝箱";break; case 63:ChestText = "珍贵的宝箱";break; } if(ChestText != undefined) Text = "[[:模板:宝箱\n]]" new mw.Api().create(self.dataPrefix() + '/point/' + id, {"bot": 1, "minor": 1, tags: 'apiedit|autoedit|aboutmap'}, Text ).then(function () { new mw.Api().ajax({action: "purge", titles: "Widget:Map3"}, {method: "post"}) self.markerWithInfo().push(id); showMarkInfo(id) }).catch(function (e) { if (e === 'nocreate-missing') { new mw.Api().ajax({action: "purge", titles: "Widget:Map3"}, {method: "post"}) self.markerWithInfo().push(id); showMarkInfo(id) } }); } var showMarkInfo = function (id) { self.selectedPoint(id) if (self.markerWithInfo().indexOf(id) === -1) { $("#infoPanel").html("此标记暂时还无内容和讨论 点击创建"); } else { var $iframe = $('<iframe src="about:blank" width="100%" height="100%" frameborder="0">'); var $btn = self.hasGeomanToolbar() ? $("").css({ display: "inline-block", cursor: "pointer", position: "absolute", right: "10px", background: "#ffffff88" }).html("编辑") : ""; $("#infoPanel").html($btn).append($iframe); $.get(`/${self.site()}/${self.dataPrefix()}/point/` + id, function (html) { $iframe[0].contentWindow.document.write(reformatInfoPgae(html)); }) } if(!self.isPhone()) self.menuTab('infoPanel') } self.editPointPage = function () { var id = self.selectedPoint(); window.open(`/${self.site()}/${self.dataPrefix()}/point/${id}?action=edit`); return ; var $iframe = $('<iframe src="about:blank" width="100%" height="100%" frameborder="0">'); $("#infoPanel").html($iframe); $.get(`/${self.site()}/${self.dataPrefix()}/point/${id}?action=edit`, function (html) { $iframe[0].contentWindow.document.write(reformatInfoPgae(html)); }) } self.saveGeoJSON = function () { util.loading(); var saveName = $("#saveName").val() var saveSummary = $("#saveSummary").val() if (saveName === "") { util.loadend(); alert("请填写保存路径名!") return false; } if (saveSummary === "") { util.loadend(); alert("请填写保存摘要!") return false; } new mw.Api().edit( self.dataPrefix() + '/' + saveName, function () { return { text: self.editingGeoJSON(), summary: saveSummary, minor: true, tags: 'apiedit|aboutmap' }; } ).then(function () { new mw.Api().ajax({action: "purge", titles: self.dataPrefix() + '/' + saveName}, {method: "post"}) .then(function () { new mw.Api().ajax({action: "purge", titles: self.dataPrefix() + '/area.json'}, {method: "post"}) new mw.Api().ajax({action: "purge", titles: self.dataPrefix() + '/route.json'}, {method: "post"}) }); self.markerOperationList.push(`保存 ${saveName} 数据成功!`); alert('Saved!'); util.loadend(); }).catch(function (e) { if (e === 'nocreate-missing') { new mw.Api().create(self.dataPrefix() + '/' + saveName, { summary: '加油!打路人', contentmodel: "text", tags: 'apiedit|aboutmap' }, "").then( self.saveGeoJSON ) } }); } self.savePoints = function () { util.loading(); var saveSummary = $("#saveSummary").val() if (saveSummary === "") { util.loadend(); alert("请填写点位保存摘要!") return false; } var type = self.selectedType(); var points = self.markersData()[type] var submitData = [], submitText = []; new mw.Api().edit( self.dataPrefix() + '/type/' + type + '/json', function (revision) { submitData = JSON.parse(revision.content || "[]") $.each(points, function (i, p) { if (!p.state) return true; if (p.state === "create") { delete p.state submitData.push(p); } else { var point = getObjectWithAttr(submitData, 'id', p.id) if (point.index === false || p.state === "fromDB") { delete p.state submitData.push(p); } else if (p.state === "remove") { submitData.splice(point.index, 1) } else if (p.state === "edit") { delete p.state submitData.splice(point.index, 1, p) } } }) $.each(submitData, function (i, p) { submitText.push(JSON.stringify(p)) }) return { text: "[" + submitText.join(",\n") + "]", summary: saveSummary, minor: true, tags: 'apiedit|aboutmap' }; } ).then(function () { new mw.Api().ajax({action: "purge", titles: self.dataPrefix() + '/type/' + type + '/json|' + self.dataPrefix() + '/type/' + type }, {method: "post"}) .then(function () { new mw.Api().ajax({action: "purge", titles: getPurgeTitles()}, {method: "post"}) }); self.markerOperationList.push(`保存 ${type} ${self.selectedTypeData().typeName} 数据成功!`); $.get(self.site() + "/" + self.dataPrefix() + '/type' + type + '/json') alert('Saved!'); $.each(submitData, function (i, p) { if (p.state) delete p.state }) updateObject(self.markersData, type, {[type]: submitData}) util.loadend() }).catch(function (e) { if (e === 'nocreate-missing') { new mw.Api().create(self.dataPrefix() + '/type/' + type, {summary: '加油!打点人', tags: 'apiedit|autoedit|aboutmap'}, "[[:模板:PointGrid]]").then(function () { setTimeout(function () { util.loadend(); new mw.Api().create(self.dataPrefix() + '/type/' + type + '/json', { summary: '加油!打点人', contentmodel: "text", tags: 'apiedit|autoedit|aboutmap' }, "").then(self.savePoints) }, 1000) }) } }); } self.getMarkerList = function () { mapApi.getOrderData(self.setMarkerLayer) } self.setMarkerLayer = function (data) { self.categoryData(data) layers['marked'] = L.canvasIconLayer({}).addTo(map); layers['marked']._canvas.setAttribute("name", 'marked') layers['marked']._canvas.style.opacity = 0.5 layers['marker'] = L.canvasIconLayer({}).addTo(map); $.each(data, function (i, order) { $.each(order.data, function (j, category) { markerTypes.push(category.markType); icons[category.markType] = L.icon({ iconUrl: category.icon, iconAnchor: category.iconAnchor || [14.5, 46], popupAnchor: category.popupAnchor || [0, -40], iconSize: category.iconSize || [29, 46], typeName: category.markTypeName, type: category.markType }); }) }) // getMarkers(markerTypes.join(",")) var dataCount = 0; if(self.dataList().length === 0 ) return dataLoaded(); mapApi.getPointData(self.dataList() , function (data) { self.markersData($.extend({}, self.markersData(), data)) setCategoryObject(); dataCount++; if(dataCount > self.dataList().length - 1) dataLoaded(); }) } self.saveUserData = function() { new mw.Api().postWithToken( 'csrf', { action: 'edit', title: "User:" + UserStatus.userInfo.mid + "/"+ storageKey +".json", text: JSON.stringify(self.markedMarker()), tags: 'apiedit|aboutmap', minor: 1 }).then(function (data) { if(data && data.edit && data.edit.result === "Success"){ alert("数据成功保存") } else { alert("数据保存失败,请刷新后尝试") } }).catch(function (e) { console.log(e) }); } self.loadUserData = function() { $.get("/ys/" + "User:" + UserStatus.userInfo.mid + "/"+ storageKey +".json?action=raw&ctype=text/json") .then(function(data){ self.markedMarker(JSON.parse(data)); localStorage.setItem(storageKey, JSON.stringify(self.markedMarker())) alert("云端数据成功同步"); setCategoryObject(); self.updateCollections(); }).catch(function(e){ if(UserStatus.userInfo.mid=== null )alert("云端数据获取失败,请检查登录状态!") else alert("未获取到云端数据。") }) } self.refreshUserData = function(){ localStorage.removeItem(storageKey); dataLoaded(); setCategoryObject(); self.updateCollections(); alert("标记数据恢复完毕,若未恢复正常请联系技术支持"); } var dataLoaded = function(){ try{ // 处理点位缓存,兼容旧版 wikiYsOpacData storageKey = "wiki" + self.game().toUpperCase() + self.dataPrefix().replace("Data:", "") + "Data", oldStorageKey = "wikiYsOpacData", self.storageKey = storageKey markedMarker = localStorage.getItem(storageKey); if(markedMarker !== null) { markedMarker = JSON.parse(markedMarker); } else if(self.game() === "ys" && self.dataPrefix() === "Data:Map" && localStorage.getItem(oldStorageKey) !== null) { markedMarker = JSON.parse(localStorage.getItem(oldStorageKey)); } else { markedMarker = [] } if($.type(markedMarker) === "array" || !markedMarker.version) { var temp = {version:"3.0", data:{}, self: {}} $.each(markedMarker, function(i, id){ var point = getPointById(id); if(!point) return true; var obj = temp.data; if(!obj[point.markType]) obj[point.markType] = {}; obj[point.markType][id] = 0; }) markedMarker = temp; } self.markedMarker(markedMarker) }catch(e){ console.log(e) } $.each(self.categoryData(), function(j, category){ $.each(category.data, function(i, type){ if(type.defaultShow === 1) self.setMarkers({markType: type.markType}) }) }) } var getPointById = function(id){ var data = self.markersData(); for(var i in data){ for(var j in data[i]){ if(data[i][j].id === id) return data[i][j]; } } } var onMarkerPopupOpen = function () { // return false; var marker = this, marked = this.options.marked, icon = this.options.icon.options self.selectPointObj(this.options); showMarkInfo(this.options.id) console.log(window.temp = this) var popupContent = $("<div style='margin-bottom: 10px;min-width: 150px;'>").html("<b>" + (this.options.title && this.options.title !== "" ? this.options.title : icon.typeName) + "</b><br>" + (this.options.title !== this.options.id ? "id: " + this.options.id  :"" ) + "<br>").prepend($("").text(!marked ? "标记" : "取消").click(function () { markMarker(this, marker); })) var $target = !self.isPhone() ? $(marker._popup._contentNode) : $("#marker-info-panel") $target.html(popupContent); this._popup.update() } var onMarkerPopupClose = function () { $("#infoPanel").html(""); self.menuTab('listPanel') self.selectedPoint(null) if(!self.isPhone()) $("#infoPanel").parents(".swipe-panel").css('top', '100%') } var markMarker = function (btn, marker) { console.log(marker) var marked = marker.options.marked ^= 1, id = marker.options.id; if(marked) { // 正在标记 $(btn).text("取消"); marker["removeFrom"](layers['marker']); marker["addTo"](layers['marked']); } else { // 正在取消标记 $(btn).text("标记"); marker["addTo"](layers['marker']); marker["removeFrom"](layers['marked']); } checkMarked(marker.options.icon.options.type, id, marked); self.updateCollections(); } var checkMarked = function(type, id, marked) { var markedMarker = self.markedMarker(), obj = markedMarker.data; if(!obj[type]) { obj[type] = {}; self.markedMarker(markedMarker); }; if(marked !== undefined) { marked ? obj[type][id] = (new Date).getTime(): delete obj[type][id]; self.markedMarker(markedMarker) localStorage.setItem(storageKey, JSON.stringify(self.markedMarker())) }; return obj[type][id] !== undefined; } self.removeAllMarkers = function () { if (layers['edit']) layers['edit'].clearLayers(); $.each(self.categorySelected(), function (type, bool) { if (!bool) return true; self.categorySelected(Object.assign({}, self.categorySelected(), {[type]: false})) }) layers['marker'].removeFrom(map); layers['marked'].removeFrom(map); layers['marked'] = L.canvasIconLayer({}).addTo(map); layers['marked']._canvas.setAttribute("name", 'marked') layers['marked']._canvas.style.opacity = 0.5 layers['marker'] = L.canvasIconLayer({}).addTo(map); } self.gotoMarker = function () { var thisMarker = this $.each(markerGroups[this.markType], function (i, marker) { if (marker.options.id === thisMarker.id) { console.log(marker) console.log(thisMarker.point) //map.setView(marker._latlng, 7) marker.openPopup() map.panTo(marker._latlng); // map.panBy(new L.Point(-200, 0)); return false; } }) } var setMarkersCanvas = function (type, mode) { // mode: true no setCategoryObject if (typeof type === "object") type = type.markType var isSelected = self.categorySelected()[type] if (!isSelected) { markerGroups[type] = []; markedGroups[type] = []; var time1 = new Date().getTime(), time2 = 0; console.log("开始 " + ((time2 = new Date().getTime()) - time1)) time1 = time2 $.each(self.areaMarkersData()[type], function (i, p) { if (p.state === "remove") return true; var marked = checkMarked(type, p.id) ? 1 : 0; var marker = L.marker([p.point.lat, p.point.lng], {icon: icons[p.markType], id: p.id, marked: marked, title: p.title}) marker.bindPopup().on("popupopen", onMarkerPopupOpen).on("popupclose", onMarkerPopupClose) if (!marked) { markerGroups[type].push(marker); } else { // marker["addTo"](layers['marked']); markedGroups[type].push(marker); } }) console.log("array " + ((time2 = new Date().getTime()) - time1)) time1 = time2 if(markerGroups[type].length > 0 ) { layers['marker'].addMarkers(markerGroups[type]) } if(markedGroups[type].length > 0 ) { layers['marked'].addMarkers(markedGroups[type]) } console.log("显示 " + ((time2 = new Date().getTime()) - time1)) time1 = time2 } else { $.each(markedGroups[type], function(i, marker) { marker["removeFrom"](layers['marked']); }) $.each(markerGroups[type], function(i, marker) { marker["removeFrom"](layers['marker']); }) } self.categorySelected(Object.assign({}, self.categorySelected(), {[type]: !isSelected})) if(!mode) setCategoryObject() if(typeof self.afterSetMarkers === "function") self.afterSetMarkers(); } self.stateColor = function (state) { if (!state) return {}; if (state === "create") return {"background": "#a9e58c"} if (state === "edit") return {"background": "#fff6aa"} if (state === "remove") return {"background": "#ffb6b6"} } var updateObject = function (object, key, data) { object(Object.assign({}, object(), {[key]: null})) object(Object.assign({}, object(), data)) } var getObjectWithAttr = function (object, attr, val) { var target = {} $.each(object, function (i, row) { if (row[attr] !== val) return true target = {index: i, row: row} return false }) return target; } var markerEditController = function (e) { if(self.selectedType() === null) { self.editingGeoJSON(self.generateGeoJson()); return false; } self.menuTab('toolPanel'); var action = e.type.replace("pm:", ""), marker = ""; if (action === "create") { marker = e.marker; } else if (action === "edit") { marker = e.sourceTarget.dragging._marker; } else if (action === "remove") { marker = e.layer || e.sourceTarget; } var id = marker.options.id, icon = marker.options.icon.options, type = marker.options.icon.options.type, data = self.markersData()[type], rowIndex = action === "create" ? -1 : data.findIndex(function (value){ return value['id'] === id }), thisMarker = { point: util.coordFormatter(marker._latlng), state: action, uid: mw.config.values.wgUserName, time: new Date().getTime() }; self.markerOperationList.push(`${action} ${type} ${icon.typeName} ${id}`) if (action === "create") { data.push(Object.assign({ "id": id, "title": "", "markType": type }, thisMarker)) } else if (action === "edit") { if (data[rowIndex].state === "create" || data[rowIndex].state === "fromDB") thisMarker.state = data[rowIndex].state Object.assign(data[rowIndex], thisMarker) } else if (action === "remove") { if (data[rowIndex].state === "create") data.splice(rowIndex) else Object.assign(data[rowIndex], thisMarker) } updateObject(self.markersData, type, {[type]: data}) } var refreshCursorMarker = function (type) { if (!type) { map.pm.Draw.Marker.options.markerStyle.id = util.getUID(); } else { map.pm.Draw.Marker.options = {markerStyle: {icon: icons[type], id: util.getUID()}}; } if (map.pm.Draw.Marker._enabled) { map.pm.disableDraw('Marker', {}); map.pm.enableDraw('Marker', {}); } } var addGeoJson = function (type){ try{ $.get(`/ys/Data:Map/route/${type.markTypeName}.json?action=raw`).then(function(data){ var layer = L.geoJSON(JSON.parse(data),{ style: function (feature) { return {color: feature.properties.color || "#3388ff"}; } }).addTo(map); layer.on('pm:markerdragend', markerEditController); }) } catch(e){console.log(e)} } var setMarkersEditable = function (type, mode) { // mode: true no setCategoryObject if (type.geojson) addGeoJson(type); if (typeof type === "object") type = type.markType if (!self.markersData()[type]) { updateObject(self.markersData, type, {[type]: []}) } var isSelected = self.categorySelected()[type] self.selectedType(!isSelected ? type : null) self.selectedTypeData(icons[type].options) if (layers['edit']) { layers['edit'].clearLayers() self.categorySelected({[type]: !isSelected}) } if (!isSelected) { refreshCursorMarker(type); markerGroups[type] = []; $.each(self.markersData()[type], function (i, p) { if (p.state === "remove") return true; markerGroups[type].push(L.marker([p.point.lat, p.point.lng], { icon: icons[p.markType], id: p.id, title: p.title }).bindPopup().on("popupopen", onMarkerPopupOpen)) }) layers['edit'] = L.layerGroup(markerGroups[type]); map.addLayer(layers['edit']); /* 补充编辑层的编辑和移除处理 */ layers['edit'].on('pm:edit', markerEditController); layers['edit'].on('pm:remove', markerEditController); } else { layers['edit'].clearLayers() map.pm.disableDraw('Marker', {}); } self.categorySelected(Object.assign({}, self.categorySelected(), {[type]: !isSelected})) if(!mode) setCategoryObject() } self.setMarkers = setMarkersCanvas /* 地图初始化相关 */ self.map = { baiduCRS: new L.Proj.CRS('EPSG:900913', '+proj=merc +a=6378206 +b=6356584.314245179 +lat_ts=0.0 +lon_0=0.0 +x_0=0 +y_0=0 +k=1.0 +units=m +nadgrids=@null +wktext +no_defs', { resolutions: function () { var level = 19, res = []; for (var i = 0; i < level; i++) { res[i] = Math.pow(2, (18 - i)) } return res; }(), origin: [0, 0] }), TileLayerCustom: L.TileLayer.extend({ getTileUrl: mapData.getTileUrl }), init: function () { map = L.map(self.mapid(), { crs: this.baiduCRS, center: self.mapCenter(), zoom: self.zoom(), maxZoom: self.maxZoom(), minZoom: self.minZoom(), maxBounds: [[86, -140],[-86, 140]], zoomControl: false }); new this.TileLayerCustom().addTo(map); L.control.zoom({zoomInTitle: '放大', zoomOutTitle: '缩小', position: 'bottomright'}).addTo(map); map.on('pm:create', function (e) { var layer = e.layer; markerEditController(e) refreshCursorMarker() layer.on('pm:edit', markerEditController); layer.on('pm:remove', markerEditController); }); map.on('pm:edit', markerEditController); map.on('pm:remove', markerEditController); /*map.on('zoomstart', function(event){ console.log(event.target); console.log(event.target._animateToZoom); //setMarkers(202) }) map.on('zoom', function(event){ //setMarkers(202) console.log(event.target); console.log(event.target._animateToZoom); }) map.on('zoomend', function(event){ console.log(event.target); console.log(event.target._animateToZoom); })*/ $("#map-div").on("click", ".simple-folder", function(){ $(this).attr("data-expand", $(this).attr("data-expand") !== "true") }) } } }; </script>
<style> #map-wrap .map-setting-btn { zoom: 0.8; font-size: 8px; width: 32px; height: 32px; } </style> <style> .leaflet-container { background: rgb(23,75,86)!important; } .leaflet-control-attribution {display:none} .leaflet-left { transition: all 0.5s; } .leaflet-popup-content-wrapper {border-radius:0} .leaflet-bar a, .leaflet-bar a:hover { background: rgba(0, 0, 0, 0.8); color: #CED3DE; border: 0; } .leaflet-bar a:hover { background: rgba(33, 33, 33, 0.8); } .leaflet-pm-toolbar .control-icon { filter: invert() brightness(1.25); } #map-wrap { border: 1px solid rgba(0, 0, 0, 0.1) } </style> <script name="map.main.js"> var map; // 地图对象 var mapApi = new BwikiMapApi(); // 地图api var mapModel, self; var mapData = { getStorageKey : function(){ return "wiki" + mapModel.game().toUpperCase() + mapModel.dataPrefix().replace("Data:", "") }, getIconUrl : function (icon) { return `https://prod-patch-wiki.biligame.com/res/ys/map/icons/${icon}?v=${mapData.default.srcVersion}` }, getCenter: function () { // #接参数可以走缓存,而?会慢一些 var center = (util.getUrlValue('hash', 'center').replace(/ /g, '') || '').split(',') return center.length === 2 ? center : [0, 0] }, getCategoryData: function () { var selectedType = localStorage.getItem(mapData.getStorageKey() + "SelectedType"); if(selectedType !== null) selectedType = selectedType.split(",") var categoryDataJson = $("#mapData #categoryData").text().trim(); if (categoryDataJson !== "") { categoryDataJson = JSON.parse(categoryDataJson) var categoryTemp = {}, categoryList = [], categoryData = [] $.each(categoryDataJson, function (i, v) { v.icon = v.icon.indexOf("http") === 0 ? v.icon : mapData.getIconUrl(v.icon); if (!categoryTemp[v.type]) { categoryList.push(v.type) categoryTemp[v.type] = {name: v.type, data: []} } if(selectedType !== null) { v.defaultShow = selectedType.indexOf(v.markType + "") !== -1?1:0 } categoryTemp[v.type].data.push(v) }) $.each(categoryList, function (i, v) { categoryData[i] = categoryTemp[v] }) return categoryData } return [] }, setCategoryData: function () { var data = mapData.getCategoryData(); data.length > 0 ? mapModel.setMarkerLayer(data) : mapModel.getMarkerList(); }, setMarkerWithInfo: function () { var data = $("#mapData #markerWithInfo").text().trim().replace(RegExp(`${mapModel.dataPrefix()}\\/point\\/`,"g"), "").split(", "); mapModel.markerWithInfo(data); }, getTileUrl: function(coords) { coords = function (coords) { var y = -coords.y + -1, r = Math.ceil((1 << coords.z - 1)/2); return ( -r / 2 <= coords.x && coords.x < r && -r <= y && y <= r / 2 - 1 && {x: coords.x, y: -coords.y + -1, z: coords.z}) }(coords); return coords ? `https://wiki-dev-patch-oss.oss-cn-hangzhou.aliyuncs.com/res/ys/map2.6-u-1/tiles/${coords.z}/tile-${coords.x}_${coords.y}.png?v=${mapModel.srcVersion()}` : `https://prod-patch-wiki.biligame.com/res/ys/map/tiles/default.png?v=${mapModel.srcVersion()}`; }, setGeoData: function () { mapModel.areaData(JSON.parse($("#mapData #mapAreaData").text().trim())); // mapModel.routeData(JSON.parse($("#mapData #mapAreaData").text().trim())); }, afterSetMarkers: function(){ var selectedType = []; $.each(mapModel.categorySelected(), function(type, selected){ if(selected) selectedType.push(type) }) localStorage.setItem(mapData.getStorageKey() + "SelectedType", selectedType.join(",")) } } var $ready = function () { $.fn.swipe = swipe; $("#map-wrap").swipe(); mapApi.set({game: 'ys', site: mw.config.get("wgGameName"), dataPrefix: 'Data:Map'}) ko.applyBindings(mapModel = new MapModel(mapData.default = { game: 'ys', site: mw.config.get("wgGameName"), mapid: 'map3', dataPrefix: 'Data:Map', mapCenter: [-25.456657, -33.90622], srcVersion: '3.0.13', zoom: 6, minZoom: 4, maxZoom: 8, isPhone: window.innerWidth < 540, toolbox: mw.config.get("wgUserGroups").indexOf("sysop") !== -1, dataList: ["point.json", "point2.json"] })); self = mapModel; mapModel.map.init(); mapData.setCategoryData(); mapData.setMarkerWithInfo(); mapData.setGeoData(); mapModel.afterSetMarkers = mapData.afterSetMarkers window.onresize=function(){ if(window.innerWidth < 540 && !mapModel.isPhone()) return mapModel.isPhone(true) if(window.innerWidth >= 540 && mapModel.isPhone()) return mapModel.isPhone(false) } }; </script> <script>(function () {var t = function () {window.jQuery && window.mw ? mw.loader.load('ext.gadget.Map3') : window.setTimeout(t, 100); }; t(); })();</script>
<script>!function(n){"use strict";function t(n,t){var r=(65535&n)+(65535&t);return(n>>16)+(t>>16)+(r>>16)<<16|65535&r}function r(n,t){return n<<t|n>>>32-t}function e(n,e,o,u,c,f){return t(r(t(t(e,n),t(u,f)),c),o)}function o(n,t,r,o,u,c,f){return e(t&r|~t&o,n,t,u,c,f)}function u(n,t,r,o,u,c,f){return e(t&o|r&~o,n,t,u,c,f)}function c(n,t,r,o,u,c,f){return e(t^r^o,n,t,u,c,f)}function f(n,t,r,o,u,c,f){return e(r^(t|~o),n,t,u,c,f)}function i(n,r){n[r>>5]|=128<<r%32,n[14+(r+64>>>9<<4)]=r;var e,i,a,d,h,l=1732584193,g=-271733879,v=-1732584194,m=271733878;for(e=0;e<n.length;e+=16)i=l,a=g,d=v,h=m,g=f(g=f(g=f(g=f(g=c(g=c(g=c(g=c(g=u(g=u(g=u(g=u(g=o(g=o(g=o(g=o(g,v=o(v,m=o(m,l=o(l,g,v,m,n[e],7,-680876936),g,v,n[e+1],12,-389564586),l,g,n[e+2],17,606105819),m,l,n[e+3],22,-1044525330),v=o(v,m=o(m,l=o(l,g,v,m,n[e+4],7,-176418897),g,v,n[e+5],12,1200080426),l,g,n[e+6],17,-1473231341),m,l,n[e+7],22,-45705983),v=o(v,m=o(m,l=o(l,g,v,m,n[e+8],7,1770035416),g,v,n[e+9],12,-1958414417),l,g,n[e+10],17,-42063),m,l,n[e+11],22,-1990404162),v=o(v,m=o(m,l=o(l,g,v,m,n[e+12],7,1804603682),g,v,n[e+13],12,-40341101),l,g,n[e+14],17,-1502002290),m,l,n[e+15],22,1236535329),v=u(v,m=u(m,l=u(l,g,v,m,n[e+1],5,-165796510),g,v,n[e+6],9,-1069501632),l,g,n[e+11],14,643717713),m,l,n[e],20,-373897302),v=u(v,m=u(m,l=u(l,g,v,m,n[e+5],5,-701558691),g,v,n[e+10],9,38016083),l,g,n[e+15],14,-660478335),m,l,n[e+4],20,-405537848),v=u(v,m=u(m,l=u(l,g,v,m,n[e+9],5,568446438),g,v,n[e+14],9,-1019803690),l,g,n[e+3],14,-187363961),m,l,n[e+8],20,1163531501),v=u(v,m=u(m,l=u(l,g,v,m,n[e+13],5,-1444681467),g,v,n[e+2],9,-51403784),l,g,n[e+7],14,1735328473),m,l,n[e+12],20,-1926607734),v=c(v,m=c(m,l=c(l,g,v,m,n[e+5],4,-378558),g,v,n[e+8],11,-2022574463),l,g,n[e+11],16,1839030562),m,l,n[e+14],23,-35309556),v=c(v,m=c(m,l=c(l,g,v,m,n[e+1],4,-1530992060),g,v,n[e+4],11,1272893353),l,g,n[e+7],16,-155497632),m,l,n[e+10],23,-1094730640),v=c(v,m=c(m,l=c(l,g,v,m,n[e+13],4,681279174),g,v,n[e],11,-358537222),l,g,n[e+3],16,-722521979),m,l,n[e+6],23,76029189),v=c(v,m=c(m,l=c(l,g,v,m,n[e+9],4,-640364487),g,v,n[e+12],11,-421815835),l,g,n[e+15],16,530742520),m,l,n[e+2],23,-995338651),v=f(v,m=f(m,l=f(l,g,v,m,n[e],6,-198630844),g,v,n[e+7],10,1126891415),l,g,n[e+14],15,-1416354905),m,l,n[e+5],21,-57434055),v=f(v,m=f(m,l=f(l,g,v,m,n[e+12],6,1700485571),g,v,n[e+3],10,-1894986606),l,g,n[e+10],15,-1051523),m,l,n[e+1],21,-2054922799),v=f(v,m=f(m,l=f(l,g,v,m,n[e+8],6,1873313359),g,v,n[e+15],10,-30611744),l,g,n[e+6],15,-1560198380),m,l,n[e+13],21,1309151649),v=f(v,m=f(m,l=f(l,g,v,m,n[e+4],6,-145523070),g,v,n[e+11],10,-1120210379),l,g,n[e+2],15,718787259),m,l,n[e+9],21,-343485551),l=t(l,i),g=t(g,a),v=t(v,d),m=t(m,h);return[l,g,v,m]}function a(n){var t,r="",e=32*n.length;for(t=0;t<e;t+=8)r+=String.fromCharCode(n[t>>5]>>>t%32&255);return r}function d(n){var t,r=[];for(r[(n.length>>2)-1]=void 0,t=0;t<r.length;t+=1)r[t]=0;var e=8*n.length;for(t=0;t<e;t+=8)r[t>>5]|=(255&n.charCodeAt(t/8))<<t%32;return r}function h(n){return a(i(d(n),8*n.length))}function l(n,t){var r,e,o=d(n),u=[],c=[];for(u[15]=c[15]=void 0,o.length>16&&(o=i(o,8*n.length)),r=0;r<16;r+=1)u[r]=909522486^o[r],c[r]=1549556828^o[r];return e=i(u.concat(d(t)),512+8*t.length),a(i(c.concat(e),640))}function g(n){var t,r,e="";for(r=0;r<n.length;r+=1)t=n.charCodeAt(r),e+="0123456789abcdef".charAt(t>>>4&15)+"0123456789abcdef".charAt(15&t);return e}function v(n){return unescape(encodeURIComponent(n))}function m(n){return h(v(n))}function p(n){return g(m(n))}function s(n,t){return l(v(n),v(t))}function C(n,t){return g(s(n,t))}function A(n,t,r){return t?r?s(t,n):C(t,n):r?m(n):p(n)}"function"==typeof define&&define.amd?define(function(){return A}):"object"==typeof module&&module.exports?module.exports=A:n.md5=A}(this); //# sourceMappingURL=md5.min.js.map</script>


Map3.0Gadget.xml

包含如下页面:

然后添加下内容至MediaWiki:Gadgets-definition页面:

* Map3[ResourceLoader|rights=edit]|leaflet.css|leaflet.js|leaflet.canvas-markers.js|leaflet.pm.css|leaflet.pm.min.js|proj4.js|proj4leaflet.min.js|Md5.min.js|Map3.js

您必须在目标Wiki上拥有管理员界面管理员权限才能执行以上两步操作。

第二步:替换底图

为地图添加底图:在widget:Map3/main中寻找方法getTileUrl,将return coords后的链接修改成地图瓦片的保存路径

getTileUrl: function(coords) {
				coords = function (coords) {
					var y = -coords.y + -1, r = Math.ceil((1 << coords.z - 4)/4 * 10);
					return ( -r <= coords.x && coords.x <= r  && -r <= y && y <= r && 
						{x: coords.x, y: -coords.y + -1, z: coords.z})
				}(coords);
				return coords ? `https://prod-patch-wiki.biligame.com/res/ys/map2/tiles/${coords.z}/tile-${coords.x}_${coords.y}.png?v=${mapModel.srcVersion()}` :
					`https://prod-patch-wiki.biligame.com/res/ys/map/tiles/default.png?v=${mapModel.srcVersion()}`;
			},			
			setGeoData: function () {
				mapModel.areaData(JSON.parse($("#mapData #mapAreaData").text().trim()));
				// mapModel.routeData(JSON.parse($("#mapData #mapAreaData").text().trim()));
			},

例如:

原神的瓦片地址为:https://prod-patch-wiki.biligame.com/res/ys/map2/

只修改路径其他的不要改

如何获得瓦片地图

  • 使用瓦片地图制作工具即可,稍后提供一个工具地址。

瓦片地图如何存放

第三步:创建点位

创建地图的对应点位信息。

第四步:输出地图

原神地图为例

<!--地图必须结构勿动-->
<div id="mapData" style="display:none">
	<div id="categoryData">{{:Data:Map/type/json}}</div>
	<div id="markerWithInfo">{{#ask:[[~Data:Map/point/*]][[!~*/json]]|link=none|limit=1000|searchlabel=}}, {{#ask:[[~Data:Map/point/*]][[!~*/json]]|link=none|limit=1000|offset=1000|searchlabel=}}</div>
<!--以下部分为地图分区功能,如果不需要分区可以删掉-->
	<div id="mapAreaData">
		[
			{"name":"提瓦特", "icon":"world", "inside":"", "geojson": null, "center": null},
			{"name":"蒙德", "icon":"mengde", "inside":"蒙德", "geojson": {{:Data:Map/蒙德.json}}, "center": [19.88871, 4.312327]},
			{"name":"璃月", "icon":"liyue", "inside":"璃月", "geojson": {{:Data:Map/璃月.json}}, "center": [-23.182214, -20.965561]},
			{"name":"雪山", "icon":"xueshan", "inside":"雪山", "geojson": {{:Data:Map/雪山.json}}, "center": [2.301491, 0.448894]},
			{"name":"稻妻", "icon":"daoqi", "inside":"稻妻", "geojson": {{:Data:Map/稻妻.json}}, "center": [-51.085105, 38.289228]}
		]
	</div><!--分区结束-->
</div>
<div>
{{#widget:Map3/css}}
{{#widget:Map3/icon}}
{{#widget:Map3/mapc}}
{{#widget:Map3/main}}
{{#widget:Md5}}
<div>
<!--地图end-->

widget:Map3/mapc页面中如下代码替换图片链接即可。

			<div class="mapMenu">
				<div class="wiki-logo">
					<img src="https://patchwiki.biligame.com/resources/assets/images/logo/logo_ys.png?t=2021052411&v=41" alt="">
				</div>

如何打点

  • 在第三步要定义好所有需要打的点
  • 打点教程:Map教程

分区

  • 划分区域(如果需要分区功能):教程暂缺

作者

迦洛翎


标签

leaflet JAVASCRIPT SMW JQUERY HTML

依赖页面

框架(Widget)

框架(Gadget)

内容

已使用该功能的页面

更新日志

  • 从3.0版本进行的重构
    • 重构底层代码
    • 重构打点功能
    • 新增点位分区
    • 新增点位标记状态储存功能