Zuletzt aktiv 4 days ago

基于 Sun-Panel 原作者"红烧猎人"提供的标题插件进行优化和美化,使其与天气插件的显示效果完全统一,实现完美的一体化体验。

wojack's Avatar wojack hat die Gist bearbeitet 4 days ago. Zu Änderung gehen

Keine Änderungen

wojack's Avatar wojack hat die Gist bearbeitet 4 days ago. Zu Änderung gehen

1 file changed, 5 insertions

使用方法.md(Datei erstellt)

@@ -0,0 +1,5 @@
1 + 直接将代码添加到 Sun-Panel 项目中即可,插件会自动:
2 + (具体添加方法请参考原作者“红烧猎人”的“[sun-panel-js-plugins](https://github.com/hslr-s/sun-panel-js-plugins)”项目)
3 + * 识别页面中的分组标题
4 + * 生成对应的导航菜单
5 + * 适配PC和移动端显示

wojack's Avatar wojack hat die Gist bearbeitet 4 days ago. Zu Änderung gehen

Keine Änderungen

wojack's Avatar wojack hat die Gist bearbeitet 4 days ago. Zu Änderung gehen

Keine Änderungen

wojack's Avatar wojack hat die Gist bearbeitet 4 days ago. Zu Änderung gehen

1 file changed, 411 insertions

toc.js(Datei erstellt)

@@ -0,0 +1,411 @@
1 + (function () {
2 + // =========== Config Start ===========
3 + // ------------------------------------
4 + // 距离滚动偏移量
5 + const scrollOffset = 80
6 +
7 + // 显示风格( auto:自动(默认) | mobile:左上角显示触发按钮-移动端风格 | sidebar:常态显示侧栏)
8 + const displayStyle = 'auto'
9 +
10 + // 移动端宽度定义
11 + const mobileWidth = 800
12 +
13 + const SunPanelTOCDomIdName = 'sun-panel-toc-dom'
14 +
15 + // 左上角按钮 SVG 图标
16 + const svgTocMobileBtn = '<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24"><path fill="currentColor" d="M17.5 4.5c-1.95 0-4.05.4-5.5 1.5c-1.45-1.1-3.55-1.5-5.5-1.5c-1.45 0-2.99.22-4.28.79C1.49 5.62 1 6.33 1 7.14v11.28c0 1.3 1.22 2.26 2.48 1.94c.98-.25 2.02-.36 3.02-.36c1.56 0 3.22.26 4.56.92c.6.3 1.28.3 1.87 0c1.34-.67 3-.92 4.56-.92c1 0 2.04.11 3.02.36c1.26.33 2.48-.63 2.48-1.94V7.14c0-.81-.49-1.52-1.22-1.85c-1.28-.57-2.82-.79-4.27-.79M21 17.23c0 .63-.58 1.09-1.2.98c-.75-.14-1.53-.2-2.3-.2c-1.7 0-4.15.65-5.5 1.5V8c1.35-.85 3.8-1.5 5.5-1.5c.92 0 1.83.09 2.7.28c.46.1.8.51.8.98z"/><path fill="currentColor" d="M13.98 11.01c-.32 0-.61-.2-.71-.52c-.13-.39.09-.82.48-.94c1.54-.5 3.53-.66 5.36-.45c.41.05.71.42.66.83s-.42.71-.83.66c-1.62-.19-3.39-.04-4.73.39c-.08.01-.16.03-.23.03m0 2.66c-.32 0-.61-.2-.71-.52c-.13-.39.09-.82.48-.94c1.53-.5 3.53-.66 5.36-.45c.41.05.71.42.66.83s-.42.71-.83.66c-1.62-.19-3.39-.04-4.73.39a1 1 0 0 1-.23.03m0 2.66c-.32 0-.61-.2-.71-.52c-.13-.39.09-.82.48-.94c1.53-.5 3.53-.66 5.36-.45c.41.05.71.42.66.83s-.42.7-.83.66c-1.62-.19-3.39-.04-4.73.39a1 1 0 0 1-.23.03"/></svg>'
17 +
18 + // ------------------------------------
19 + // =========== Config End ===========
20 +
21 + // 滚动容器的类名
22 + const scrollContainerElementClassName = '.scroll-container'
23 +
24 + // 一些函数
25 + const isMobile = () => {
26 + if (displayStyle === 'mobile') {
27 + return true
28 + }
29 + else if (displayStyle === 'pc') {
30 + return false
31 + }
32 + const width = window.innerWidth
33 + return width < mobileWidth
34 + }
35 +
36 + function createDom() {
37 + // 检测是否已经存在TOC DOM,存在则删除
38 + (function () {
39 + const element = document.getElementById(SunPanelTOCDomIdName)
40 + if (element) {
41 + element.remove()
42 + }
43 + })()
44 +
45 + const SunPanelTOCDom = document.createElement('div')
46 + SunPanelTOCDom.id = SunPanelTOCDomIdName
47 + document.body.appendChild(SunPanelTOCDom)
48 +
49 + // ========= Add style start =========
50 + const style = document.createElement('style')
51 + const SunPanelTOCDomStyleId = `#${SunPanelTOCDomIdName}`
52 + style.textContent = `
53 + ${SunPanelTOCDomStyleId} #toc-mobile-btn {
54 + top: 20px !important;
55 + left: 20px !important;
56 + position: fixed;
57 + width: 44px;
58 + height: 44px;
59 + background: rgba(255, 255, 255, 0.15);
60 + backdrop-filter: blur(10px);
61 + border-radius: 12px;
62 + display: flex;
63 + justify-content: center;
64 + align-items: center;
65 + cursor: pointer;
66 + box-shadow: 0 4px 12px rgba(0, 0, 0, 0.15);
67 + border: 1px solid rgba(255, 255, 255, 0.2);
68 + z-index: 9998;
69 + transition: all 0.3s ease;
70 + }
71 +
72 + ${SunPanelTOCDomStyleId} #toc-mobile-btn:hover {
73 + transform: scale(1.05);
74 + background: rgba(255, 255, 255, 0.25);
75 + }
76 +
77 + ${SunPanelTOCDomStyleId} .hidden {
78 + display: none !important;
79 + }
80 +
81 + ${SunPanelTOCDomStyleId} #toc-sidebar {
82 + width: 20px;
83 + padding: 10px;
84 + position: fixed;
85 + top: 0;
86 + left: 0;
87 + height: 100%;
88 + overflow: hidden;
89 + display: flex;
90 + flex-direction: column;
91 + justify-content: center;
92 + transition: width 0.3s ease, background-color 0.3s ease;
93 + border-top-right-radius: 16px;
94 + border-bottom-right-radius: 16px;
95 + background: rgba(255, 255, 255, 0.15);
96 + backdrop-filter: blur(15px);
97 + border: 1px solid rgba(255, 255, 255, 0.2);
98 + box-shadow: 0 8px 32px rgba(0, 0, 0, 0.1);
99 + z-index: 9999;
100 + }
101 +
102 + ${SunPanelTOCDomStyleId} .toc-mobile-btn-svg-container{
103 + width:21px;
104 + height:21px;
105 + color: rgba(255, 255, 255, 0.9);
106 + }
107 +
108 + ${SunPanelTOCDomStyleId} .toc-sidebar-expansion {
109 + width: 180px !important;
110 + display: flex;
111 + background: rgba(255, 255, 255, 0.15);
112 + backdrop-filter: blur(15px);
113 + box-shadow: 0 8px 32px rgba(0, 0, 0, 0.1);
114 + }
115 +
116 + ${SunPanelTOCDomStyleId} #toc-sidebar .toc-sidebar-box {
117 + width: 100%;
118 + padding: 20px 10px;
119 + }
120 +
121 + ${SunPanelTOCDomStyleId} .title-bar-box {
122 + display: flex;
123 + align-items: center;
124 + position: relative;
125 + cursor: pointer;
126 + margin-bottom: 15px;
127 + padding: 8px 12px;
128 + border-radius: 8px;
129 + transition: all 0.3s ease;
130 + }
131 +
132 + ${SunPanelTOCDomStyleId} .title-bar-box:hover {
133 + background: rgba(255, 255, 255, 0.1);
134 + }
135 +
136 + ${SunPanelTOCDomStyleId} .title-bar-slip {
137 + width: 20px;
138 + height: 6px;
139 + background-color: rgba(255, 255, 255, 0.8);
140 + border-radius: 4px;
141 + transition: height 0.3s ease, width 0.3s ease;
142 + }
143 +
144 + ${SunPanelTOCDomStyleId} .title-bar-title {
145 + opacity: 0;
146 + white-space: nowrap;
147 + transition: opacity 0.3s ease, transform 0.3s ease, margin-left 0.3s ease;
148 + font-size: 14px;
149 + color: white;
150 + font-weight: 500;
151 + }
152 +
153 + ${SunPanelTOCDomStyleId} .toc-sidebar-expansion .title-bar-title {
154 + opacity: 1;
155 + margin-left: 12px;
156 + }
157 +
158 + ${SunPanelTOCDomStyleId} .toc-sidebar-expansion .title-bar-box:hover .title-bar-slip {
159 + width: 30px;
160 + background-color: rgba(255, 255, 255, 1);
161 + }
162 +
163 + ${SunPanelTOCDomStyleId} .toc-sidebar-expansion .title-bar-box:hover .title-bar-title {
164 + transform: translateX(5px);
165 + color: rgba(255, 255, 255, 1);
166 + }
167 +
168 + /* 移动端侧边栏样式 */
169 + ${SunPanelTOCDomStyleId} .toc-mobile-sidebar {
170 + width: 180px !important;
171 + height: 100%;
172 + border-radius: 0 16px 16px 0;
173 + background: rgba(255, 255, 255, 0.15);
174 + backdrop-filter: blur(15px);
175 + z-index: 10000;
176 + }
177 +
178 + ${SunPanelTOCDomStyleId} .toc-close-btn {
179 + position: absolute;
180 + top: 20px;
181 + right: 20px;
182 + width: 44px;
183 + height: 44px;
184 + background: rgba(255, 255, 255, 0.15);
185 + backdrop-filter: blur(10px);
186 + border-radius: 12px;
187 + display: flex;
188 + justify-content: center;
189 + align-items: center;
190 + cursor: pointer;
191 + box-shadow: 0 4px 12px rgba(0, 0, 0, 0.15);
192 + border: 1px solid rgba(255, 255, 255, 0.2);
193 + z-index: 10001;
194 + transition: all 0.3s ease;
195 + }
196 +
197 + ${SunPanelTOCDomStyleId} .toc-close-btn:hover {
198 + transform: scale(1.05);
199 + background: rgba(255, 255, 255, 0.25);
200 + }
201 +
202 + ${SunPanelTOCDomStyleId} .toc-close-btn svg {
203 + width: 20px;
204 + height: 20px;
205 + color: rgba(255, 255, 255, 0.9);
206 + }
207 +
208 + `
209 + // 添加样式到文档头部
210 + SunPanelTOCDom.appendChild(style)
211 +
212 + // ========= Add style end =========
213 +
214 + // 添加移动端菜单按钮
215 + const tocMobileBtn = document.createElement('div')
216 + tocMobileBtn.id = 'toc-mobile-btn'
217 + SunPanelTOCDom.appendChild(tocMobileBtn)
218 +
219 + const tocMobileBtnSvgcContainer = document.createElement('div')
220 + tocMobileBtnSvgcContainer.innerHTML = svgTocMobileBtn
221 + tocMobileBtnSvgcContainer.classList.add('toc-mobile-btn-svg-container')
222 + tocMobileBtn.appendChild(tocMobileBtnSvgcContainer)
223 +
224 + // 创建侧边栏容器
225 + const sidebar = document.createElement('div')
226 + sidebar.id = 'toc-sidebar'
227 +
228 + const sidebarBox = document.createElement('div')
229 + sidebarBox.className = 'toc-sidebar-box'
230 +
231 + // 查询出所有类名包含 item-group-index- 的元素
232 + const items = document.querySelectorAll('[class*="item-group-index-"]')
233 +
234 + // 遍历并打印每个元素的完整类名
235 + items.forEach((item) => {
236 + item.classList.forEach((className) => {
237 + if (className.startsWith('item-group-index-')) {
238 + const titleBarBox = document.createElement('div')
239 + titleBarBox.className = 'title-bar-box'
240 + titleBarBox.dataset.groupClassName = className
241 +
242 + // 目录条
243 + const titleBarSlip = document.createElement('div')
244 + titleBarSlip.className = 'title-bar-slip'
245 +
246 + // 创建一个链接
247 + const titleBarTitle = document.createElement('div')
248 + titleBarTitle.className = 'title-bar-title'
249 +
250 + // 获取子元素中 class="group-title" 的内容
251 + const titleElement = item.querySelector('.group-title')
252 + const titleText = titleElement ? titleElement.textContent : item.id
253 + titleBarTitle.textContent = titleText
254 +
255 + titleBarBox.appendChild(titleBarSlip)
256 + titleBarBox.appendChild(titleBarTitle)
257 +
258 + sidebarBox.appendChild(titleBarBox)
259 + }
260 + })
261 + })
262 +
263 + sidebar.appendChild(sidebarBox)
264 +
265 + // 添加关闭按钮(移动端使用)
266 + const closeBtn = document.createElement('div')
267 + closeBtn.className = 'toc-close-btn hidden'
268 + closeBtn.innerHTML = '<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24"><path fill="currentColor" d="M19 6.41L17.59 5 12 10.59 6.41 5 5 6.41 10.59 12 5 17.59 6.41 19 12 13.41 17.59 19 19 17.59 13.41 12z"/></svg>'
269 + closeBtn.addEventListener('click', (e) => {
270 + e.stopPropagation()
271 + mobileHideSidebar()
272 + })
273 + SunPanelTOCDom.appendChild(closeBtn)
274 +
275 + // 将侧边栏添加到页面中
276 + SunPanelTOCDom.appendChild(sidebar)
277 +
278 + function mobileHideSidebar() {
279 + sidebar.classList.remove('toc-sidebar-expansion', 'toc-mobile-sidebar')
280 + sidebar.classList.add('hidden')
281 + closeBtn.classList.add('hidden')
282 + }
283 +
284 + function hideSidebar() {
285 + sidebar.classList.remove('toc-sidebar-expansion')
286 + }
287 +
288 + function showSidebar() {
289 + sidebar.classList.add('toc-sidebar-expansion')
290 + sidebar.classList.remove('hidden')
291 +
292 + // 移动端显示关闭按钮
293 + if (isMobile()) {
294 + sidebar.classList.add('toc-mobile-sidebar')
295 + closeBtn.classList.remove('hidden')
296 + } else {
297 + sidebar.classList.remove('toc-mobile-sidebar')
298 + closeBtn.classList.add('hidden')
299 + }
300 + }
301 +
302 + // ----------------
303 + // 监听宽度变化开始
304 + // ----------------
305 + function debounce(func, wait) {
306 + let timeout
307 + return function (...args) {
308 + clearTimeout(timeout)
309 + timeout = setTimeout(() => {
310 + func.apply(this, args)
311 + }, wait)
312 + }
313 + }
314 +
315 + function handleResize() {
316 + if (isMobile()) {
317 + tocMobileBtn.classList.remove('hidden')
318 + sidebar.classList.add('hidden')
319 + closeBtn.classList.add('hidden')
320 + }
321 + else {
322 + tocMobileBtn.classList.add('hidden')
323 + sidebar.classList.remove('hidden')
324 + closeBtn.classList.add('hidden')
325 + }
326 + }
327 +
328 + // 使用防抖函数包装你的处理函数
329 + const debouncedHandleResize = debounce(handleResize, 200)
330 +
331 + // 添加事件监听器
332 + window.addEventListener('resize', debouncedHandleResize)
333 +
334 + // 首次触发
335 + handleResize()
336 +
337 + // ----------------
338 + // 监听宽度变化结束
339 + // ----------------
340 +
341 + // 监听移动端按钮点击
342 + tocMobileBtn.addEventListener('click', () => {
343 + if (sidebar.classList.contains('toc-sidebar-expansion')) {
344 + // 隐藏
345 + mobileHideSidebar()
346 + }
347 + else {
348 + // 显示
349 + showSidebar()
350 + }
351 + })
352 +
353 + // 监听TOC栏失去hover
354 + sidebar.addEventListener('mouseleave', () => {
355 + if (!isMobile()) {
356 + hideSidebar()
357 + }
358 + })
359 +
360 + // 监听TOC栏获得hover
361 + sidebar.addEventListener('mouseenter', () => {
362 + if (!isMobile()) {
363 + showSidebar()
364 + }
365 + })
366 +
367 + // 监听TOC点击事件
368 + document.querySelectorAll('.title-bar-box').forEach((box) => {
369 + box.addEventListener('click', function (event) {
370 + // 检查触发事件的元素是否有 'data-groupClassName' 属性
371 + if (this.dataset.groupClassName) {
372 + // 获取 'data-groupClass' 属性的值
373 + const groupClassName = this.dataset.groupClassName
374 + // 使用属性值作为选择器查询对应的元素
375 + const targetElement = document.querySelector(`.${groupClassName}`)
376 + if (targetElement) {
377 + // 获取目标元素的 'top' 坐标
378 + const targetTop = targetElement.offsetTop
379 + const scrollContainerElement = document.querySelector(scrollContainerElementClassName)
380 + if (scrollContainerElement) {
381 + scrollContainerElement.scrollTo({
382 + top: targetTop - scrollOffset,
383 + behavior: 'smooth', // 平滑滚动
384 + })
385 + }
386 + }
387 + }
388 +
389 + // 移动端点击后自动关闭侧边栏
390 + if (isMobile()) {
391 + mobileHideSidebar()
392 + }
393 + })
394 + })
395 + }
396 +
397 + // 判断是否已经存在分组,不存在将定时监听
398 + const items = document.querySelectorAll('[class*="item-group-index-"]')
399 + if (items.length > 0) {
400 + createDom()
401 + return
402 + }
403 +
404 + const interval = setInterval(() => {
405 + const items = document.querySelectorAll('[class*="item-group-index-"]')
406 + if (items.length > 0) {
407 + createDom()
408 + clearInterval(interval)
409 + }
410 + }, 1000)
411 + })()
Neuer Älter