الاطّلاع على عدة طرق لإضافة تأثيرات متحركة إلى الحدود في CSS
ضبط الحدود
هناك بضع طرق متاحة لضبط حدود على عنصر: border
وoutline
وbox-shadow
. كما هو موضّح بالتفصيل في مقالة The 3 CSS Methods for Adding Element Borders (الثلاث طرق لاستخدام CSS لإضافة حدود العناصر) التي كتبتها "ستيفاني إكليس"، تتمتع كل طريقة بمزاياها وعيوبها، خاصةً عندما يتعلق الأمر بإضافة تأثيرات متحركة إلى الحدود. السبب الرئيسي لعدم استخدام border
CSS مناسب هو لأغراض متعلّقة بالحركة.
من المقالات التي لفتت انتباهي مؤخرًا هي Fantastic CSS border animation (تأثيرات حدود رائعة باستخدام CSS)، حيث استكشافت المؤلّفة "كوكو" المزيد من الخيارات. من خلال إدخال محتوى تم إنشاؤه باستخدام ::before
و::after
، يتم إنشاء حدود زائفة يتم تحريكها بعد ذلك.
ما يلفت انتباهي أكثر هو الرسومات البيانية المتحركة الداعمة المستخدَمة في المقالة. وتساعد هذه المراجع في شرح الإجراءات التي يتم اتّخاذها لتحقيق التأثير المطلوب.
إنّ كلّ من الطبقة البيضاء والخطوط الملونة هي محتوى من إنشاء المستخدم. من خلال تمويه الطبقة البيضاء وإظهارها، يصبح من الواضح كيفية تداخلها وطريقة عمل الصورة المتحركة.
الاحتفاظ بنموذج العلبة
من عيوب استخدام محتوى من إنشاء المستخدمين لمحاكاة حدود أنّك تنتهي بـ نموذج مربّع غير مكتمل: يمكن أن يحجب المحتوى الآن الحدود الزائفة لأنّه تمّ رسم "الحدود" المذكورة تحت المحتوى. للحدّ من هذه المشكلة، عليك تطبيق border-width
المطلوب كpadding
.
للحصول على حدود حقيقية، وبالتالي الاحتفاظ بوظائف نموذج المربّع، يمكنك استخدام خلفيات متعدّدة ثم تمديدها إلى منطقة الحدود.
الأساسيات
لنبدأ بإنشاء حدود منقطة وإضافة الخلفيات المتعددة.
/* Size of the border */
--border-size: 0.5rem;
/* Create a dotted border */
border: var(--border-size) dotted lime;
/* Create two background layers:
1. A white semi-transparent
2. A layer with the colored boxes
*/
background-image:
linear-gradient(to right, rgb(255 255 255 / 0.5), rgb(255 255 255 / 0.5)),
conic-gradient(
from 45deg,
#d53e33 0deg 90deg,
#fbb300 90deg 180deg,
#377af5 180deg 270deg,
#399953 270deg 360deg
)
;
ضبط حجم الخلفيات باستخدام background-origin
كما ترى، هناك مشكلة في الخلفيات هنا: تمّ رسمها في الحدود، ولكن يبدو أنّ conic-gradient
غير صحيحة. هذا السلوك هو السلوك المطلوب: لا يتم رسم صور الخلفية تلقائيًا في الحدود لأنّ مصدرها هو padding-box
للعنصر. لإنشاء حدود، يتم تكرار صور الخلفية المحدّدة في الحدود نفسها، ما يؤدي إلى ظهور التأثير المرئي الغريب.
لحلّ هذه المشكلة، عليك تمديد الخلفية لكي تشغل أيضًا حجم الحدود. يمكنك إجراء ذلك يدويًا من خلال تمديد الخلفية وإعادة وضعها، ولكن من الأفضل استخدام السمة background-origin
لضبط حجم الخلفية وفقًا للعنصر border-box
.
/* Manually add or offset the size of the border where needed */
background-position: calc(var(--border-size) * -1) calc(var(--border-size) * -1);
background-size: calc(var(--border-size) * 2 + 100%) calc(var(--border-size) * 2 + 100%);
background-origin: border-box;
تؤدي هذه الإضافة إلى تحسين الأداء بشكل كبير:
تصغير طبقة الخلفية البيضاء باستخدام background-clip
بما أنّ الخلفيات تشغل الآن كل المساحة، يجب تصغير الطبقة شبه الشفافة مرة أخرى. بدلاً من تعديل background-size
مرة أخرى، هناك طريقة أسهل لإجراء ذلك: استخدِم background-clip
واضبطه على padding-box
. بهذه الطريقة، لن يتم رسم الخلفية تحت منطقة الحدود.
background-clip:
padding-box, /* Clip white semi-transparent to the padding-box */
border-box /* Clip colored boxes to the border-box (default) */
;
أخيرًا، اضبط الحد على transparent
للحصول على التأثير الكامل.
border: 0.3rem dotted transparent;
Animation
لاستعادة الحركة في الحدود، يمكنك التحكّم في زاوية البدء للعنصر conic-gradient
.
--angle: 0deg;
conic-gradient(
from var(--angle),
#d53e33 0deg 90deg,
#fbb300 90deg 180deg,
#377af5 180deg 270deg,
#399953 270deg 360deg
);
بفضل @property، يمكنك إجراء ذلك بسهولة في المتصفّحات المتوافقة:
@property --angle {
syntax: "<angle>";
initial-value: 0deg;
inherits: false;
}
@keyframes rotate {
to {
--angle: 360deg;
}
}
بعد دمج كل هذه العناصر، تصبح التعليمة البرمجية على النحو التالي:
المحتوى الإضافي: border-image
من الطرق التي سبق أن شرحناها لرسم حدود متدرّجة هي استخدام CSS border-image
.
ويتيح ذلك استخدام رمز أبسط لأنّك لست بحاجة إلى التعامل مع الخلفيات المتداخلة. يمكن تطبيق الصور المتحركة بالطريقة نفسها كما في السابق.
/* Create a border */
border: 0.5rem solid transparent;
/* Paint an image in the border */
border-image:
conic-gradient(
from var(--angle),
#d53e33 0deg 90deg,
#fbb300 90deg 180deg,
#377af5 180deg 270deg,
#399953 270deg 360deg
) 1
;
ومع ذلك، ستلاحظ أنّ بعض العناصر لم تعُد تعمل مع هذا النهج:
- لا يتّبع
border-image
border-radius
، بل سيظل مستطيلاً دائمًا. - عند ضبط
border-image-slice
على التعبئة، لا يتم رسمborder-image
أسفلbackground
المحدَّد، بل فوقه. وقد يكون ذلك مزعجًا إذا كنت تريد أن تكون الخلفية شفافة جزئيًا.
في الختام
هناك العديد من الطرق لإضافة تأثيرات متحركة إلى الحدود في CSS. ويمكنك الاعتماد على أحدهما أو الآخر حسب حالة الاستخدام.