Javascript fading - плавное изменение цвета

Источник: http://www.tigir.com/ Автор: Цыгырлаш Игорь

Как вы уже, наверное, догадались, в статье пойдет речь не о селекции, не о том как из ромашки вывести яблоню (так что юным Мичуринцам не сюда :-), а речь пойдет об изменении цвета элементов веб-страницы, о плавном перетекании одного цвета в другой, т.е. поговорим об fade-эффектах.

Пример

Начнем с примера: наведите курсор на рисунок, а затем уберите курсор.

<!--[if lt IE 5.5]> <![endif]-->
Fading

Если вас не интересуют теоретические подробности, а нужно готовое решение то вам сюда.

Постановка задачи

Дано два цвета: начальный цвет и конечный цвет.

Необходимо найти промежуточные цвета, применяя которые поочередно к элементу на странице, мы получим плавный переход от начального цвета к конечному. А так же следует ещё выяснить, сколько должно быть этих промежуточных цветов и через какое время должна происходить смена одного промежуточного цвета другим.

Взгляд вглубь вопроса

Возьмем в качестве начального цвета, например, белый (white), а в качестве конечного цвета оранжево-красный (orangered).

#FFFFFF ? ... n ... ? #FF4500

Теперь необходимо найти промежуточные цвета. Хорошо. Но как?! С какой стороны подойти к этому вопросу? Для этого вспомним (или узнаем :-), как формируется цвет на экране монитора. Любой цвет на экране монитора формируется из трех основных цветов: красного (Red), зеленого (Green) и синего (Blue), путем их смешения (т.е. используется цветовая модель RGB). А указываются цвета на веб-странице либо численными значениями в той же системе RGB, либо литералами именованных цветов (например, White для белого, Red для красного и т.д., однако не у всех цветов есть имена), которые, все равно, указывают на численные значения. Но рассматривать задание цвета по имени не будем, ибо имена придуманы для удобства запоминания человеком, но в нашем случае они создадут неудобства при вычислениях, т.к. все равно потребуют перевода в численную форму. Задать численное значение цвета можно двумя способами: шестнадцатеричным и функциональным.

  • В шестнадцатеричном представлении запись значения RGB имеет следующий формат: символ '#', непосредственно за которым следует три или шесть шестнадцатеричных символов. Значение RGB из трех цифр (#rgb) преобразуется в последовательность из шести цифр (#rrggbb) путем дублирования цифр, а не добавления нулей. Например, #fb0 расширяется до #ffbb00. Поэтому белый цвет (#ffffff) можно указать в более короткой форме (#fff).
  • В функциональном представлении формат записи значения RGB имеет следующий вид: строка 'rgb(', непосредственно за которой следует список из трех разделенных запятыми вещественных (или целочисленных, или процентных) значений, непосредственно за которыми следует скобка ')'. Целочисленное значение 255 эквивалентно процентному значению 100% и шестнадцатеричным значениям F или FF, так что rgb(255,255,255) = rgb(100%,100%,100%) = #FFF.

Таким образом, цвет, заданный в числовой форме дает нам значения своих составляющих цветов, что, в конечном итоге, дает нам возможность, изменяя каждый из основных цветов первоначального цвета прийти ко второму, конечному цвету. Если рассматривать наш конкретный пример, то имеем следующее (в скобках указаны значения в десятичной системе):

#FFFFFF = FF (255) FF (255) FF (255)
0 – CE (206) – FF (255)
#FF4500 = FF (255) 45 (49) 0

Т.е. для того чтобы из белого цвета получить оранжево-красный нужно красную составляющую белого цвета оставить без изменения (изменить величину на ноль), от зеленой вычесть 206, а из синей вычесть 255. Назовем эти числа (ΔR = 0, ΔG = -206, ΔB= -255) приращениями.

Теперь, чтобы получить, например, два промежуточных цвета + конечный цвет (всего 3), нужно изменять начальные значения RGB-триплета (#FFFFFF) не на полную величину приращений ΔR, ΔG, ΔB, а сначала на 1/3, потом на 2/3 и напоследок на 3/3 (3/3 = 1, это полное значение приращения для получения конечного цвета, который нам, в принципе, и так известен).

#FFFFFF = FF (255) FF (255) FF (255)
#FFBAAA = 255 -0 255 - 206*1/3 = 186 (BA) 255 - 255*1/3 = 170 (AA)
#FF7655 = 255 - 0 255 - 206*2/3 = 118 (76) 255 - 255*2/3 = 85 (55)
#FF4500 = FF (255) 45 (49) 0

Итак, вычислять промежуточные цвета мы научились. Осталось прояснить ещё один нюанс, для того чтобы человек мог заметить промежуточные цвета, нужно делать временные задержки между сменой цветов, иначе изменения произойдут так быстро, что пользователь просто не заметит промежуточных цветов.

Теперь у нас есть ясная общая картина и мы можем перейти к написанию кода. Есть начальный и конечный цвета, мы можем вычислить n промежуточных цветов (где n выбираем произвольно), и есть величина задержки t (где t выбираем произвольно). Таким образом алгоритм таков: назначаем элементу на веб-странице первый промежуточный цвет, делаем задержку на величину t, назначаем элементу второй промежуточный цвет, делаем задержку, ..., назначаем элементу n-ый промежуточный цвет, который является конечным цветом.

Реализация

В качестве примера, сделаем кнопку, у которой при клике меняется фон с белого на оранжево-красный.

Поехали... Первая неправильная мысль, которая может прийти в голову, это в цикле вычислять промежуточные цвета, делая задержку после каждой итерации.

...

function fade()
{
  for (var i = 1; i <= n; i++)
  {
    //присвоить в качестве фонового цвета кнопки i-тый цвет
    кнопка.style.backgroundColor = текущий_промежуточный_цвет( i );
    задержка(t);
  }
}

...

Почему неправильная? Потому что, приостановить выполнения кода Javascript невозможно! Однако можно сделать эмуляцию задержки используя методы setInterval или setTimeout. Эти методы выполняют код Javascript с задержкой на заданное количество миллисекунд (setInterval выполняет код многократно через заданный интервал времени, а setTimeout один раз по истечении заданного времени). Верхний цикл можно представить так:

var i = 0; //индекс текущего промежуточного цвета

function fade()
{
  i++; //изменяем индекс промежуточного цвета
  
  //присвоить в качестве фонового цвета кнопки i-тый цвет
  кнопка.style.backgroundColor = текущий_промежуточный_цвет( i );
  
  /* если ещё не перебраны все промежуточные цвета,
  то вызвать fade() через t миллисекунд опять */
  if ( i < n ) setInterval(fade, t); 
}

...

Полный пример выглядит так:

 
 

  


 

   

 

Надеюсь, что вы поняли суть процесса, т.к. вышеприведенный пример больше подходит для обучения, нежели для реального использования. Этот пример рассчитан на фейдинг одного объекта в одном направлении, т.к. жестко заданны параметры в виде глобальных переменных и id элемента. Как реализовать независимые процессы для разных элементов? Как прервать процесс фейдинга и вернуться, так же плавно, к исходному цвету?

Ниже будет представлен законченный с функциональной точки зрения код, решающий поставленные проблемы, но сейчас я хочу немного прояснить ответ на второй вопрос. Как вернуться от промежуточного цвета, до которого успел дойти процесс, к начальному, должно быть понятно. Но как остановить уже запущенный процесс? Как вы помните, процесс реализуется посредством вызова метода setTimeout, поэтому ответ на вопрос заключается в отмене метода setTimeout. При вызове setTimeout() возвращает значение, которое является указателем на этот "таймаут", и чтобы отменить его, нужно вызвать метод clearTimeout(), в качестве параметра которому передается указатель на "таймаут", возвращенный методом setTimeout.

Готовое решение

Библиотека fade.js. Пример её использования ниже.



  
  
  
   <!-- Подключаем библиотеку fade.js -->
  
  
  
  


  
  • Fade
  • Fade
  • Fade

Основные шаги:

  1. Подключаем библиотеку функций;
  2. Определяем правила;
  3. Вызываем метод fade() для перетекания цвета от начального к конечному, или fade.back() для возврата к начальному цвету.

Разжевываем

Как подключать библиотеку, думаю, видно из примера выше. Теперь стоит пояснить определение правил. До того как вызвать фейдинг нужно определить правила, по которым он будет выполняться: нужно определить начальный и конечный цвета, так же нужно определить сколько нужно генерировать промежуточных цветов между начальным и конечным; нужно знать к какому CSS свойству элемента применять генерируемые цвета, а также какую временную задержку делать перед сменой одного промежуточного цвета на другой.

Правила определяются с помощью метода fade.addRule

Синтаксис: fade.addRule (sRuleName, sFadeStartColor, sFadeFinishColor, sCSSProp, nMiddleColors, nDelay)

Аргументы:

  • sRuleName - имя правила, задаётся произвольно;
  • sFadeStartColor и sFadeFinishColor - начальный и конечный цвета заданные в шестнадцатиричном виде (полном или сокращенном) ;
  • sCSSProp - CSS свойство цвета которое будет изменяться;
  • nMiddleColors - количество промежуточных цветов (необязательный аргумент, по умолчанию равен 50) ;
  • nDelay - задержка между сменой промежуточных цветов в миллисекундах (необязательный аргумент, по умолчанию равен 1).

Сам вызов фейдинга делаем через методы fade(sElemId, sRule) и fade.back(sElemId, sRule), где sElemId это id элемента, а sRule список правил разделенных запятой.



Опубликовал admin
17 Мар, Суббота 2007г.



Программирование для чайников.