بیاموزید که چگونه Local Font Access API به شما امکان می دهد به فونت های نصب شده محلی کاربر دسترسی داشته باشید و جزئیات سطح پایینی در مورد آنها بدست آورید.
فونت های ایمن وب
اگر به اندازه کافی توسعه وب را انجام داده اید، ممکن است فونت های به اصطلاح امن وب را به خاطر بیاورید. این فونتها تقریباً در تمام نمونههای پرکاربردترین سیستمعاملها (یعنی Windows، macOS، رایجترین توزیعهای لینوکس، اندروید و iOS) در دسترس هستند. در اوایل دهه 2000، مایکروسافت حتی ابتکاری به نام فونتهای اصلی TrueType برای وب را رهبری کرد که این فونتها را به صورت رایگان برای دانلود ارائه میکرد با این هدف که «هر وقت از وبسایتی بازدید میکنید که آنها را مشخص میکند، صفحات را دقیقاً همانطور که طراح سایت در نظر داشته است ببینید. " بله، این شامل سایتهایی است که در Comic Sans MS تنظیم شدهاند. در اینجا یک پشته فونت ایمن وب کلاسیک (با آخرین فونت sans-serif
) ممکن است شبیه به این باشد:
body {
font-family: Helvetica, Arial, sans-serif;
}
فونت های وب
روزهایی که فونت های ایمن وب واقعا اهمیت داشتند، مدت هاست گذشته است. امروزه، فونتهای وب داریم، که برخی از آنها حتی فونتهای متغیری هستند که میتوانیم با تغییر مقادیر برای محورهای مختلف در معرض آن، آنها را تغییر دهیم. میتوانید با اعلام یک بلوک @font-face
در ابتدای CSS از فونتهای وب استفاده کنید، که فایل(های) فونت را برای دانلود مشخص میکند:
@font-face {
font-family: 'FlamboyantSansSerif';
src: url('flamboyant.woff2');
}
پس از این، میتوانید از فونت وب سفارشی با تعیین font-family
استفاده کنید.
body {
font-family: 'FlamboyantSansSerif';
}
فونت های محلی به عنوان وکتور اثر انگشت
بیشتر فونت های وب از وب می آیند. با این حال، یک واقعیت جالب این است که ویژگی src
در اعلان @font-face
، به غیر از تابع url()
، یک تابع local()
را نیز می پذیرد. این اجازه می دهد تا فونت های سفارشی بارگذاری شوند (تعجب!) به صورت محلی. اگر کاربر FlamboyantSansSerif را بر روی سیستم عامل خود نصب کرده باشد، از نسخه محلی به جای دانلود استفاده می شود:
@font-face {
font-family: 'FlamboyantSansSerif';
src: local('FlamboyantSansSerif'), url('flamboyant.woff2');
}
این رویکرد مکانیزم بازگشتی خوبی را ارائه می دهد که به طور بالقوه باعث صرفه جویی در پهنای باند می شود. در اینترنت، متأسفانه، ما نمی توانیم چیزهای خوبی داشته باشیم. مشکل تابع local()
این است که می توان از آن برای انگشت نگاری مرورگر سوء استفاده کرد. به نظر می رسد، لیست فونت هایی که کاربر نصب کرده است می تواند بسیار قابل شناسایی باشد. بسیاری از شرکت ها فونت های شرکتی خود را دارند که روی لپ تاپ کارمندان نصب می شوند. به عنوان مثال، گوگل یک فونت شرکتی به نام Google Sans دارد.
یک مهاجم می تواند با آزمایش وجود تعداد زیادی فونت شرکتی شناخته شده مانند Google Sans ، تعیین کند که شخصی برای چه شرکتی کار می کند. مهاجم سعی میکند متنی را با این فونتها بر روی بوم رندر کند و حروف را اندازهگیری کند. اگر حروف با شکل شناخته شده فونت شرکت مطابقت داشته باشد، مهاجم ضربه می خورد. اگر حروفها مطابقت نداشته باشند، مهاجم میداند که از یک فونت جایگزین پیشفرض استفاده شده است زیرا فونت شرکتی نصب نشده است. برای جزئیات کامل در مورد این حمله و سایر حملات اثر انگشت مرورگر، مقاله نظرسنجی Laperdix و همکاران را بخوانید.
فونت های شرکت جدا از هم، حتی فقط لیست فونت های نصب شده را می توان شناسایی کرد. وضعیت این بردار حمله به قدری بد شده است که اخیراً تیم WebKit تصمیم گرفت "فقط فونتهای وب و فونتهایی را که به همراه سیستم عامل ارائه میشوند [در لیست فونتهای موجود] بگنجانند، اما فونتهای نصب شده توسط کاربر محلی را شامل نمیشود" . (و اینجا هستم، با مقاله ای در مورد اعطای دسترسی به فونت های محلی.)
Local Font Access API
شروع این مقاله ممکن است شما را در روحیه منفی قرار داده باشد. آیا واقعاً می توانیم چیزهای خوبی نداشته باشیم؟ ناراحت نباش ما فکر می کنیم که می توانیم، و شاید همه چیز ناامید کننده نباشد . اما ابتدا اجازه دهید به سوالی که ممکن است از خودتان بپرسید پاسخ دهم.
چرا وقتی فونت های وب وجود دارد به Local Font Access API نیاز داریم؟
ارائه ابزارهای گرافیکی و طراحی با کیفیت حرفه ای در طول تاریخ دشوار بوده است. یکی از موانع، ناتوانی در دسترسی و استفاده از انواع فونت های حرفه ای ساخته شده و اشاره شده است که طراحان به صورت محلی نصب کرده اند. فونتهای وب برخی موارد استفاده از انتشار را فعال میکنند، اما دسترسی برنامهریزی شده به شکلهای حروف برداری و جداول فونتهای مورد استفاده توسط شطرنجسازها را برای ارائه خطوط کلی حروف فعال نمیکنند. به همین ترتیب هیچ راهی برای دسترسی به داده های باینری فونت وب وجود ندارد.
- ابزارهای طراحی برای اجرای طرحبندی OpenType خود نیاز به دسترسی به بایتهای فونت دارند و به ابزارهای طراحی اجازه میدهند تا در سطوح پایینتر، برای اقداماتی مانند انجام فیلترهای برداری یا تبدیلها بر روی اشکال حروف، قلاب شوند.
- توسعه دهندگان ممکن است پشته های فونت قدیمی برای برنامه های خود داشته باشند که در وب می آورند. برای استفاده از این پشته ها، معمولاً نیاز به دسترسی مستقیم به داده های فونت دارند، چیزی که فونت های وب ارائه نمی دهند.
- ممکن است برخی از فونت ها برای تحویل از طریق وب مجوز نداشته باشند. برای مثال، Linotype مجوزی برای برخی از فونتها دارد که فقط شامل استفاده از دسکتاپ میشود.
Local Font Access API تلاشی برای حل این چالش هاست. از دو بخش تشکیل شده است:
- یک API شمارش فونت ، که به کاربران اجازه دسترسی به مجموعه کامل فونتهای موجود سیستم را میدهد.
- از هر نتیجه شمارش، امکان درخواست دسترسی به ظرف SFNT سطح پایین (بایت محور) که شامل داده های کامل فونت است.
پشتیبانی از مرورگر
نحوه استفاده از Local Font Access API
تشخیص ویژگی
برای بررسی اینکه آیا Local Font Access API پشتیبانی میشود، از:
if ('queryLocalFonts' in window) {
// The Local Font Access API is supported
}
شمارش فونت های محلی
برای به دست آوردن لیستی از فونت های نصب شده محلی، باید window.queryLocalFonts()
را فراخوانی کنید. اولین بار، این یک درخواست مجوز را راه اندازی می کند که کاربر می تواند آن را تأیید یا رد کند. اگر کاربر فونتهای محلی خود را تأیید کند تا درخواست شود، مرورگر آرایهای را با دادههای فونت برمیگرداند که میتوانید روی آن حلقه بزنید. هر فونت به عنوان یک شی FontData
با family
خصوصیات (مثلاً "Comic Sans MS"
)، fullName
(به عنوان مثال، "Comic Sans MS"
)، postscriptName
(به عنوان مثال، "ComicSansMS"
) و style
(مثلاً) نشان داده می شود. ، "Regular"
).
// Query for all available fonts and log metadata.
try {
const availableFonts = await window.queryLocalFonts();
for (const fontData of availableFonts) {
console.log(fontData.postscriptName);
console.log(fontData.fullName);
console.log(fontData.family);
console.log(fontData.style);
}
} catch (err) {
console.error(err.name, err.message);
}
اگر فقط به زیرمجموعهای از فونتها علاقه دارید، میتوانید با افزودن یک پارامتر postscriptNames
، آنها را بر اساس نامهای PostScript فیلتر کنید.
const availableFonts = await window.queryLocalFonts({
postscriptNames: ['Verdana', 'Verdana-Bold', 'Verdana-Italic'],
});
دسترسی به داده های SFNT
دسترسی کامل به SFNT از طریق متد blob()
شی FontData
در دسترس است. SFNT یک فرمت فایل فونتی است که میتواند حاوی فونتهای دیگری مانند فونتهای PostScript، TrueType، OpenType، Web Open Font Format (WOFF) و سایر فونتها باشد.
try {
const availableFonts = await window.queryLocalFonts({
postscriptNames: ['ComicSansMS'],
});
for (const fontData of availableFonts) {
// `blob()` returns a Blob containing valid and complete
// SFNT-wrapped font data.
const sfnt = await fontData.blob();
// Slice out only the bytes we need: the first 4 bytes are the SFNT
// version info.
// Spec: https://docs.microsoft.com/en-us/typography/opentype/spec/otff#organization-of-an-opentype-font
const sfntVersion = await sfnt.slice(0, 4).text();
let outlineFormat = 'UNKNOWN';
switch (sfntVersion) {
case '\x00\x01\x00\x00':
case 'true':
case 'typ1':
outlineFormat = 'truetype';
break;
case 'OTTO':
outlineFormat = 'cff';
break;
}
console.log('Outline format:', outlineFormat);
}
} catch (err) {
console.error(err.name, err.message);
}
نسخه ی نمایشی
میتوانید Local Font Access API را در دمو زیر مشاهده کنید. حتماً کد منبع را نیز بررسی کنید. دمو یک عنصر سفارشی به نام <font-select>
را به نمایش می گذارد که یک انتخاب کننده فونت محلی را پیاده سازی می کند.
ملاحظات حریم خصوصی
به نظر میرسد مجوز "local-fonts"
سطحی بسیار قابل اثرانگشت را فراهم میکند. با این حال، مرورگرها آزادند هر چیزی را که دوست دارند برگردانند. به عنوان مثال، مرورگرهای متمرکز بر ناشناس بودن ممکن است فقط مجموعه ای از فونت های پیش فرض را که در مرورگر تعبیه شده است، ارائه دهند. به طور مشابه، مرورگرها نیازی به ارائه داده های جدول دقیقاً همانطور که روی دیسک نشان داده می شوند، نیستند.
هرجا که ممکن است، Local Font Access API طراحی شده است تا دقیقاً اطلاعات مورد نیاز برای فعال کردن موارد استفاده ذکر شده را نشان دهد. API های سیستم ممکن است لیستی از فونت های نصب شده را نه به صورت تصادفی یا مرتب شده، بلکه به ترتیب نصب فونت تولید کنند. بازگرداندن دقیقاً فهرست فونتهای نصبشده ارائهشده توسط چنین API سیستمی میتواند دادههای اضافی را که ممکن است برای انگشت نگاری استفاده شود در معرض نمایش بگذارد، و موارد استفادهای که میخواهیم فعال کنیم با حفظ این ترتیب کمکی نمیکند. در نتیجه، این API مستلزم آن است که داده های برگشتی قبل از بازگرداندن مرتب شوند.
امنیت و مجوزها
تیم Chrome با استفاده از اصول اصلی تعریف شده در کنترل دسترسی به ویژگیهای قدرتمند پلتفرم وب ، از جمله کنترل کاربر، شفافیت و ارگونومی، Local Font Access API را طراحی و پیادهسازی کرده است.
کنترل کاربر
دسترسی به فونتهای کاربر کاملاً تحت کنترل آنها است و اجازه داده نمیشود مگر اینکه مجوز "local-fonts"
، همانطور که در رجیستری مجوز فهرست شده است، داده شود.
شفافیت
اینکه آیا به سایتی اجازه دسترسی به فونت های محلی کاربر داده شده است، در برگه اطلاعات سایت قابل مشاهده است.
تداوم مجوز
مجوز "local-fonts"
بین بارگیری مجدد صفحه باقی خواهد ماند. از طریق برگه اطلاعات سایت قابل ابطال است.
بازخورد
تیم Chrome میخواهد درباره تجربیات شما با Local Font Access API بشنود.
در مورد طراحی API به ما بگویید
آیا چیزی در مورد API وجود دارد که آنطور که انتظار داشتید کار نمی کند؟ یا آیا روش ها یا ویژگی هایی وجود دارد که برای اجرای ایده خود به آنها نیاز دارید؟ سوال یا نظری در مورد مدل امنیتی دارید؟ یک مشکل مشخصات را در مخزن GitHub مربوطه ثبت کنید یا افکار خود را به یک مشکل موجود اضافه کنید.
گزارش مشکل در اجرا
آیا اشکالی در پیاده سازی کروم پیدا کردید؟ یا پیاده سازی با مشخصات متفاوت است؟ یک اشکال را در new.crbug.com ثبت کنید. مطمئن شوید که تا جایی که می توانید جزئیات، دستورالعمل های ساده برای بازتولید را وارد کنید و Blink>Storage>FontAccess
در کادر Components وارد کنید. Glitch برای به اشتراک گذاری سریع و آسان تکرارها عالی عمل می کند.
پشتیبانی از API را نشان دهید
آیا قصد دارید از Local Font Access API استفاده کنید؟ پشتیبانی عمومی شما به تیم Chrome کمک میکند ویژگیها را اولویتبندی کند و به سایر فروشندگان مرورگر نشان میدهد که چقدر حمایت از آنها ضروری است.
با استفاده از هشتگ #LocalFontAccess
یک توییت به ChromiumDev@ ارسال کنید و به ما اطلاع دهید که کجا و چگونه از آن استفاده میکنید.
لینک های مفید
- توضیح دهنده
- پیش نویس مشخصات
- اشکال کرومیوم برای شمارش فونت
- اشکال Chromium برای دسترسی به جدول فونت
- ورودی ChromeStatus
- مخزن GitHub
- بررسی تگ
- موقعیت استاندارد موزیلا
قدردانی
مشخصات Local Font Access API توسط Emil A. Eklund ، Alex Russell ، Joshua Bell و Olivier Yiptong ویرایش شده است. این مقاله توسط Joe Medley , Dominik Röttsches و Olivier Yiptong بررسی شده است . تصویر قهرمان توسط برت جردن در Unsplash .