Pada hari lain saya sedang mengusahakan JSON Schema Generator, dan mahu memaparkan nombor baris dalam
Saya melakukan beberapa penyelidikan dan menemui pelbagai pendekatan:
Saya tidak menyukai mana-mana daripada mereka! Yang pertama tidak kelihatan segar — dan tidak sepadan dengan gaya yang telah saya sediakan untuk elemen
Yang kedua memerlukan sekumpulan JavaScript untuk mengekalkan senarai tertib itu: menambah/mengalih keluar
Jadi saya akhirnya mencipta hibrid.
Ia ialah SVG yang dijana secara dinamik, disimpan sebagai Harta Tersuai CSS — dan digunakan sebagai imej latar belakang, mewarisi gaya daripada elemen
Jom selami.
Pertama, kaedah utama:
lineNumbers(element, numLines = 50, inline = false)
elemen ialah
Seterusnya, kami mentakrifkan awalan untuk sifat tersuai:
const prefix = '--linenum-';
Sebelum kami meneruskan, kami menyemak sama ada hendak menggunakan semula mana-mana harta sedia ada:
if (!inline) { const styleString = document.body.getAttribute('style') || ''; const regex = new RegExp(`${prefix}[^:]*`, 'g'); const match = styleString.match(regex); if (match) { element.style.backgroundImage = `var(${match[0]})`; return; } }
Seterusnya, kami mengekstrak gaya daripada elemen, menjadikan SVG dengan keluarga fon, saiz fon, ketinggian garis dsb. yang sama :
const bgColor = getComputedStyle(element).borderColor; const fillColor = getComputedStyle(element).color; const fontFamily = getComputedStyle(element).fontFamily; const fontSize = parseFloat(getComputedStyle(element).fontSize); const lineHeight = parseFloat(getComputedStyle(element).lineHeight) / fontSize; const paddingTop = parseFloat(getComputedStyle(element).paddingTop) / 2; const translateY = (fontSize * lineHeight).toFixed(2);
Kami memerlukan id rawak untuk harta kami juga:
const id = `${prefix}${Math.random().toString(36).substr(2, 6)}`;
Dan kini tiba masanya untuk memaparkan SVG:
const svg = `<svg xmlns="http://www.w3.org/2000/svg"> <style> svg { background: ${bgColor}; } text { fill: hsl(from ${fillColor} h s l / 50%); font-family: ${fontFamily}; font-size: ${fontSize}px; line-height: ${lineHeight}; text-anchor: end; translate: 0 calc((var(--n) * ${translateY}px) + ${paddingTop}px); } </style> ${Array.from({ length: numLines }, (_, i) => `<text x="90%" style="--n:${i + 1};">${i + 1}</text>`).join("")} </svg>`;
Mari kita pecahkan:
Dalam bahagian