امکانات جدید در کروم 65
CSS Paint API (همچنین بهعنوان «CSS Custom Paint» یا «Houdini's paint Worklet» نیز شناخته میشود) بهطور پیشفرض با شروع Chrome 65 فعال است. چیست؟ چه کاری می توانید با آن انجام دهید؟ و چگونه کار می کند؟ خوب، ادامه دهید، آیا…
CSS Paint API به شما این امکان را می دهد که هر زمان که یک ویژگی CSS انتظار یک تصویر را داشت، به صورت برنامه نویسی یک تصویر تولید کنید. ویژگی هایی مانند background-image
یا border-image
معمولاً با url()
برای بارگذاری یک فایل تصویر یا با توابع داخلی CSS مانند linear-gradient()
استفاده می شود. به جای استفاده از آنها، اکنون میتوانید از paint(myPainter)
برای ارجاع به یک Worklet رنگ استفاده کنید.
نوشتن کارنامه رنگ
برای تعریف یک Worklet paint به نام myPainter
، باید یک فایل Worklet paint CSS را با استفاده از CSS.paintWorklet.addModule('my-paint-worklet.js')
بارگذاری کنیم. در آن فایل، میتوانیم از تابع registerPaint
برای ثبت کلاس Worklet paint استفاده کنیم:
class MyPainter {
paint(ctx, geometry, properties) {
// ...
}
}
registerPaint('myPainter', MyPainter);
در داخل callback paint()
، میتوانیم از ctx
به همان روشی که از CanvasRenderingContext2D
استفاده میکنیم، همانطور که از <canvas>
میشناسیم استفاده کنیم. اگر میدانید چگونه در یک <canvas>
طراحی کنید، میتوانید در یک صفحه رنگ نقاشی بکشید! geometry
عرض و ارتفاع بوم را که در اختیار ماست به ما می گوید. properties
در ادامه این مقاله توضیح خواهم داد.
به عنوان یک مثال مقدماتی، بیایید یک Worklet paint شطرنجی بنویسیم و از آن به عنوان تصویر پسزمینه یک <textarea>
استفاده کنیم. (من از یک textarea استفاده می کنم زیرا به طور پیش فرض قابل تغییر اندازه است.):
<!-- index.html -->
<!doctype html>
<style>
textarea {
background-image: paint(checkerboard);
}
</style>
<textarea></textarea>
<script>
CSS.paintWorklet.addModule('checkerboard.js');
</script>
// checkerboard.js
class CheckerboardPainter {
paint(ctx, geom, properties) {
// Use `ctx` as if it was a normal canvas
const colors = ['red', 'green', 'blue'];
const size = 32;
for(let y = 0; y < geom.height/size; y++) {
for(let x = 0; x < geom.width/size; x++) {
const color = colors[(x + y) % colors.length];
ctx.beginPath();
ctx.fillStyle = color;
ctx.rect(x * size, y * size, size, size);
ctx.fill();
}
}
}
}
// Register our class under a specific name
registerPaint('checkerboard', CheckerboardPainter);
اگر در گذشته از <canvas>
استفاده کرده اید، این کد باید آشنا به نظر برسد. دمو زنده را اینجا ببینید.
تفاوت با استفاده از یک تصویر پسزمینه رایج در اینجا این است که هر زمان که کاربر اندازه ناحیه متنی را تغییر دهد، الگو در صورت تقاضا دوباره ترسیم میشود. این بدان معناست که تصویر پسزمینه همیشه دقیقاً به اندازهای است که باید باشد، از جمله جبران نمایشگرهای با چگالی بالا.
این بسیار جالب است، اما همچنین کاملا ثابت است. آیا میخواهیم هر بار که الگوی مشابهی را میخواهیم اما با مربعهایی با اندازههای متفاوت، یک Worklet جدید بنویسیم؟ پاسخ منفی است!
پارامترهای کاری خود را
خوشبختانه، Worklet paint میتواند به سایر ویژگیهای CSS دسترسی داشته باشد، جایی که properties
پارامتر اضافی وارد عمل میشوند. با دادن ویژگی استاتیک inputProperties
به کلاس، می توانید در تغییرات هر ویژگی CSS، از جمله ویژگی های سفارشی، مشترک شوید. مقادیر از طریق پارامتر properties
به شما داده می شود.
<!-- index.html -->
<!doctype html>
<style>
textarea {
/* The paint worklet subscribes to changes of these custom properties. */
--checkerboard-spacing: 10;
--checkerboard-size: 32;
background-image: paint(checkerboard);
}
</style>
<textarea></textarea>
<script>
CSS.paintWorklet.addModule('checkerboard.js');
</script>
// checkerboard.js
class CheckerboardPainter {
// inputProperties returns a list of CSS properties that this paint function gets access to
static get inputProperties() { return ['--checkerboard-spacing', '--checkerboard-size']; }
paint(ctx, geom, properties) {
// Paint worklet uses CSS Typed OM to model the input values.
// As of now, they are mostly wrappers around strings,
// but will be augmented to hold more accessible data over time.
const size = parseInt(properties.get('--checkerboard-size').toString());
const spacing = parseInt(properties.get('--checkerboard-spacing').toString());
const colors = ['red', 'green', 'blue'];
for(let y = 0; y < geom.height/size; y++) {
for(let x = 0; x < geom.width/size; x++) {
ctx.fillStyle = colors[(x + y) % colors.length];
ctx.beginPath();
ctx.rect(x*(size + spacing), y*(size + spacing), size, size);
ctx.fill();
}
}
}
}
registerPaint('checkerboard', CheckerboardPainter);
اکنون میتوانیم از کد یکسانی برای انواع مختلف شطرنجها استفاده کنیم. اما حتی بهتر از آن، اکنون میتوانیم به DevTools برویم و با مقادیر کمانچه بپردازیم تا زمانی که ظاهر مناسب را پیدا کنیم.
مرورگرهایی که از paint worklet پشتیبانی نمی کنند
در زمان نگارش این مقاله، فقط Chrome Worklet رنگ را پیادهسازی کرده است. در حالی که سیگنالهای مثبتی از سایر فروشندگان مرورگر وجود دارد، پیشرفت چندانی وجود ندارد. برای به روز ماندن، بررسی کنید آیا هودینی هنوز آماده است؟ به طور منظم در عین حال، مطمئن شوید که از بهبود پیشرونده استفاده کنید تا کد خود را در حال اجرا نگه دارید، حتی اگر هیچ پشتیبانی از Worklet رنگ وجود نداشته باشد. برای اطمینان از اینکه کارها همانطور که انتظار می رود کار می کنند، باید کد خود را در دو مکان تنظیم کنید: CSS و JS.
تشخیص پشتیبانی از Worklet paint در JS می تواند با بررسی شی CSS
انجام شود: js if ('paintWorklet' in CSS) { CSS.paintWorklet.addModule('mystuff.js'); }
برای سمت CSS، دو گزینه دارید. می توانید از @supports
استفاده کنید:
@supports (background: paint(id)) {
/* ... */
}
یک ترفند فشرده تر، استفاده از این واقعیت است که CSS در صورت وجود یک تابع ناشناخته، کل یک اعلان ویژگی را باطل می کند و متعاقباً نادیده می گیرد. اگر یک ویژگی را دو بار مشخص کنید - ابتدا بدون رنگ و سپس با رنگ کار - بهبود تدریجی دریافت خواهید کرد:
textarea {
background-image: linear-gradient(0, red, blue);
background-image: paint(myGradient, red, blue);
}
در مرورگرهایی که از Worklet paint پشتیبانی میکنند، دومین اعلان background-image
مورد اول را بازنویسی میکند. در مرورگرهایی که از paint worklet پشتیبانی نمیکنند ، اعلان دوم نامعتبر است و کنار گذاشته میشود و اعلان اول به اجرا در میآید.
CSS Paint Polyfill
برای بسیاری از کاربردها، می توان از CSS Paint Polyfill نیز استفاده کرد که پشتیبانی از CSS Custom Paint و Paint Worklets را به مرورگرهای مدرن اضافه می کند.
موارد استفاده کنید
موارد استفاده زیادی برای کارگاه رنگ وجود دارد که برخی از آنها واضح تر از بقیه هستند. یکی از بارزترین موارد استفاده از رنگ کاری برای کاهش اندازه DOM است. اغلب، عناصر صرفا برای ایجاد تزئینات با استفاده از CSS اضافه می شوند. برای مثال، در Material Design Lite، دکمه با افکت ریپل حاوی 2 عنصر <span>
اضافی برای پیاده سازی خود ریپل است. اگر تعداد زیادی دکمه دارید، این میتواند به تعداد زیادی از عناصر DOM اضافه شود و منجر به کاهش عملکرد در تلفن همراه شود. اگر به جای آن افکت ریپل را با استفاده از Worklet رنگ اجرا کنید ، در نهایت با 0 عنصر اضافی و فقط یک Worklet رنگ مواجه خواهید شد. علاوه بر این، شما چیزی دارید که سفارشی کردن و پارامترسازی آن بسیار ساده تر است.
مزیت دیگر استفاده از رنگ ورکلت این است که - در اکثر سناریوها - راه حلی که از ورکلت رنگ استفاده می کند از نظر بایت کوچک است. البته، یک معامله وجود دارد: هر زمان که اندازه بوم یا هر یک از پارامترها تغییر کند، کد رنگ شما اجرا می شود. بنابراین اگر کد شما پیچیده باشد و طولانی باشد، ممکن است jank را معرفی کند. کروم در حال کار بر روی جابجایی ورکلتهای رنگ از رشته اصلی است، بهگونهای که حتی ورکلتهای رنگ طولانیمدت نیز بر روی پاسخدهی رشته اصلی تأثیری نمیگذارند.
برای من، هیجانانگیزترین چشمانداز این است که رنگ ورکلت امکان پرکردن کارآمد ویژگیهای CSS را میدهد که مرورگر هنوز از آن برخوردار نیست. یکی از مثالها این است که گرادیانهای مخروطی را چند بار پر کنید تا زمانی که به صورت بومی در کروم قرار بگیرند. مثال دیگر: در یک جلسه CSS، تصمیم گرفته شد که اکنون می توانید چندین رنگ حاشیه داشته باشید. در حالی که این جلسه هنوز ادامه داشت، همکارم Ian Kilpatrick یک polyfill برای این رفتار CSS جدید با استفاده از رنگ Worklet نوشت .
تفکر خارج از "جعبه"
اکثر مردم وقتی در مورد رنگ کاری یاد می گیرند شروع به فکر کردن به تصاویر پس زمینه و تصاویر حاشیه می کنند. یکی از موارد استفاده کمتر بصری برای Worklet رنگ mask-image
است تا عناصر DOM دارای اشکال دلخواه باشند. به عنوان مثال یک الماس :
mask-image
تصویری به اندازه عنصر می گیرد. مناطقی که تصویر ماسک شفاف است، عنصر شفاف است. مناطقی که تصویر ماسک مات است، عنصر مات است.
اکنون در کروم
Worklet Paint مدتی است که در Chrome Canary وجود دارد. با کروم 65، به طور پیش فرض فعال است. ادامه دهید و امکانات جدیدی را که Worklet رنگ باز می کند امتحان کنید و آنچه را که ساخته اید به ما نشان دهید! برای الهام بیشتر، نگاهی به مجموعه وینسنت دی اولیویرا بیندازید.