2013-10-28 120 views
4

我試圖創建一個像Windows 8安裝We're getting your PC ready一樣的連續顏色轉換。我想不出如何編寫shift函數。檢查所有R,G,B值並將當前顏色與下一種顏色相匹配。連續顏色轉換

任何人都可以幫助我嗎?或者讓我知道是否有比這更好的方法呢?

function switchColor(id) { 
    var elm = document.getElementById(id); 

    // getting elm's current rgb values 
    var elmColor = window.getComputedStyle(elm).getPropertyValue("background"); 
    var startIndex = elmColor.indexOf("("); 
    var finishIndex = elmColor.indexOf(")"); 
    var elmRGB = elmColor.substring(startIndex + 1, finishIndex); 
    var currentColor = elmRGB.split(","); 
    for (var i = 0; i<3; i++) { currentColor[i] = currentColor[i].trim(); } 

    // generating a random color => [r, g, ,b] 
    var nextColor = []; 
    for (var i = 0; i < 3; i++) { 
     nextColor[i] = Math.floor(Math.random()*250); 
    } 

    // function to convert rgb array to hex color => [r, g, b] = #rgb 
    function rgbToHex(clr) { 
     var rgb = clr; 
     var hex; 
     var hex1 = rgb[0].toString(16); 
     var hex2 = rgb[1].toString(16); 
     var hex3 = rgb[2].toString(16); 
     if (hex1.length < 2) { hex1 = "0" + hex1; } 
     if (hex2.length < 2) { hex2 = "0" + hex2; } 
     if (hex3.length < 2) { hex3 = "0" + hex3; } 
     return hex = "#" + hex1 + hex2 + hex3; 
    } 

    // checking if nextColor rgb values are greater than current rgb's 
    // so we can increase or decrease for smooth transition 
    var status = []; 
    for (var i = 0; i < 3; i++) { 
     if (nextColor[i] > currentColor[i]) { status.push(1); } 
     else { status.push(0); } 
    } 

    // this isn't part of the code, just testing 
    elm.style.background = rgbToHex(nextColor); 

    function shift() { 
     // shift between colors 
     // modify currentColor's rgb values and apply it to the elm 
     // elm.style.background = rgbToHex(currentColor); 
    } 
    var handler = setInterval(shift, 100); 
} 
setInterval(function() { switchColor("sandbox"); }, 2000); 

JSFiddle

+0

請說明您的具體問題或添加額外的細節,突顯正是你需要的。正如目前所寫,很難確切地說出你在問什麼。 – Marcin

+0

試圖更具體。它應該做的。我猜。 – akinuri

+0

注意 - 你有一個失控的'setInterval'循環,你在'switchColor'裏面啓動一個新的100ms定時器,並且永遠不會取消它 - 每個2s你再做一次... – Alnitak

回答

12

檢查此JSFiddle過渡與一個奇特的圖形。

/* ==================== Required Functions ==================== */ 
 
// This is required to get the initial background-color of an element. 
 
// The element might have it's bg-color already set before the transition. 
 
// Transition should continue/start from this color. 
 
// This will be used only once. 
 
function getElementBG(elm) { 
 
\t var bg \t = getComputedStyle(elm).backgroundColor; 
 
\t \t bg \t = bg.match(/\((.*)\)/)[1]; 
 
\t \t bg \t = bg.split(","); 
 
\t for (var i = 0; i < bg.length; i++) { 
 
\t \t bg[i] = parseInt(bg[i], 10); 
 
\t } 
 
\t if (bg.length > 3) { bg.pop(); } 
 
\t return bg; 
 
} 
 

 
// A function to generate random numbers. 
 
// Will be needed to generate random RGB value between 0-255. 
 
function random() { 
 
\t if (arguments.length > 2) { 
 
\t \t return 0; 
 
\t } 
 
\t switch (arguments.length) { 
 
\t \t case 0: 
 
\t \t \t return Math.random(); 
 
\t \t case 1: 
 
\t \t \t return Math.round(Math.random() * arguments[0]); 
 
\t \t case 2: 
 
\t \t \t var min = arguments[0]; 
 
\t \t \t var max = arguments[1]; 
 
\t \t \t return Math.round(Math.random() * (max - min) + min); 
 
\t } 
 
} 
 

 
// Generates a random RGB value. 
 
function generateRGB(min, max) { 
 
\t var min \t \t = min || 0; 
 
\t var max \t \t = min || 255; 
 
\t var color \t = []; 
 
\t for (var i = 0; i < 3; i++) { 
 
\t \t var num = random(min, max); 
 
\t \t color.push(num); 
 
\t } 
 
\t return color; 
 
} 
 

 
// Calculates the distance between the RGB values. 
 
// We need to know the distance between two colors 
 
// so that we can calculate the increment values for R, G, and B. 
 
function calculateDistance(colorArray1, colorArray2) { 
 
\t var distance = []; 
 
\t for (var i = 0; i < colorArray1.length; i++) { 
 
\t \t distance.push(Math.abs(colorArray1[i] - colorArray2[i])); 
 
\t } 
 
\t return distance; 
 
} 
 

 
// Calculates the increment values for R, G, and B using distance, fps, and duration. 
 
// This calculation can be made in many different ways. 
 
function calculateIncrement(distanceArray, fps, duration) { 
 
\t var fps \t \t \t = fps || 30; 
 
\t var duration \t = duration || 1; 
 
\t var increment \t = []; 
 
\t for (var i = 0; i < distanceArray.length; i++) { 
 
\t \t var incr = Math.abs(Math.floor(distanceArray[i]/(fps * duration))); 
 
\t \t if (incr == 0) { 
 
\t \t \t incr = 1; 
 
\t \t } 
 
\t \t increment.push(incr); 
 
\t } 
 
\t return increment; 
 
} 
 

 
// Converts RGB array [32,64,128] to HEX string #204080 
 
// It's easier to apply HEX color than RGB color. 
 
function rgb2hex(colorArray) { 
 
\t var color = []; 
 
\t for (var i = 0; i < colorArray.length; i++) { 
 
\t \t var hex = colorArray[i].toString(16); 
 
\t \t if (hex.length < 2) { hex = "0" + hex; } 
 
\t \t color.push(hex); 
 
\t } 
 
\t return "#" + color.join(""); 
 
} 
 

 
/* ==================== Setup ==================== */ 
 
// Duration is not what it says. It's a multiplier in the calculateIncrement() function. 
 
// duration = 1-4, fast-to-slow 
 
var fps \t \t \t \t = 30; 
 
var duration \t \t = 3; 
 
var transElement \t = document.body; 
 
var currentColor \t = getElementBG(transElement); 
 
var transHandler \t = null; 
 

 
startTransition(); 
 

 
/* ==================== Transition Initiator ==================== */ 
 
function startTransition() { 
 
\t clearInterval(transHandler); 
 
\t 
 
\t targetColor \t = generateRGB(); 
 
\t distance \t = calculateDistance(currentColor, targetColor); 
 
\t increment \t = calculateIncrement(distance, fps, duration); 
 
\t 
 
\t transHandler = setInterval(function() { 
 
\t \t transition(); 
 
\t }, 1000/fps); 
 
} 
 

 
/* ==================== Transition Calculator ==================== */ 
 
function transition() { 
 
\t // checking R 
 
\t if (currentColor[0] > targetColor[0]) { 
 
\t \t currentColor[0] -= increment[0]; 
 
\t \t if (currentColor[0] <= targetColor[0]) { 
 
\t \t \t increment[0] = 0; 
 
\t \t } 
 
\t } else { 
 
\t \t currentColor[0] += increment[0]; 
 
\t \t if (currentColor[0] >= targetColor[0]) { 
 
\t \t \t increment[0] = 0; 
 
\t \t } 
 
\t } 
 
\t 
 
\t // checking G 
 
\t if (currentColor[1] > targetColor[1]) { 
 
\t \t currentColor[1] -= increment[1]; 
 
\t \t if (currentColor[1] <= targetColor[1]) { 
 
\t \t \t increment[1] = 0; 
 
\t \t } 
 
\t } else { 
 
\t \t currentColor[1] += increment[1]; 
 
\t \t if (currentColor[1] >= targetColor[1]) { 
 
\t \t \t increment[1] = 0; 
 
\t \t } 
 
\t } 
 
\t 
 
\t // checking B 
 
\t if (currentColor[2] > targetColor[2]) { 
 
\t \t currentColor[2] -= increment[2]; 
 
\t \t if (currentColor[2] <= targetColor[2]) { 
 
\t \t \t increment[2] = 0; 
 
\t \t } 
 
\t } else { 
 
\t \t currentColor[2] += increment[2]; 
 
\t \t if (currentColor[2] >= targetColor[2]) { 
 
\t \t \t increment[2] = 0; 
 
\t \t } 
 
\t } 
 
\t 
 
\t // applying the new modified color 
 
\t transElement.style.backgroundColor = rgb2hex(currentColor); 
 
\t 
 
\t // transition ended. start a new one 
 
\t if (increment[0] == 0 && increment[1] == 0 && increment[2] == 0) { 
 
\t \t startTransition(); 
 
\t } 
 
}
body { 
 
    background: white; 
 
}

+0

這是一個了不起的實現。你能否幫我理解增量停車的目的是什麼?還有一點關於它是如何工作的。我試圖實現類似的東西,所以你的解釋真的會有所幫助! :) –

+0

@NehaAgrawal我試圖解釋[本頁](http://akinuri.com/exps/color-transition/)代碼中發生了什麼。如果你仍然有問題,我可以更具體。 – akinuri

+0

前一個鏈接不再有效。這裏是新的[鏈接](http://akinuri.com/exps/color-transition/intro)。 – akinuri

0

哪裏是太多的變數? 這種方式呢?

var el = document.getElementById("sandbox"), 
    interval = 2000; 

function getNewColor(){ 
    //generate color 
} 

function getOldColor(el){ 
    //get current color 
} 

function switchColor(el, oldColor, newColor){ 
    //change color 
} 

setInterval(function(){ 

    swithColors(el, getOldColor(el), getNewColor()); 

},interval); 
+0

可以說currentColor是[128,160,200],下一個顏色是[60,140,​​230]。我需要檢查currentColor [0]> nextColor [0],如果是的話,我會減少它。如果沒有,增加。我需要用所有的r,g,b值來做到這一點。而增加/減少他們需要在一個區間。每次迭代更新元素,以便平滑過渡。當我說變量太多時,我並沒有談論「變量」。我的意思是有很多事情要考慮。 – akinuri

0

感謝akinuri我設法他的答案適應上requestanimationframe運行的動態功能。再次感謝akinuri,不錯的代碼。 PS:currentColor和targetColor請求與RGB值的字符串( 'RGB(0,0,0)')

function startColorFade(fps, duration, element, currentColor, targetColor) { 
    var stop = false; 
    var fpsInterval = 1000/fps; 
    var now; 
    var then = Date.now(); 
    var elapsed; 
    var startTime = then; 
    var currentColorArray = getElementBG(currentColor); 
    var targetColorArray = getElementBG(targetColor); 
    var distance = calculateDistance(currentColorArray, targetColorArray); 
    var increment = calculateIncrement(distance, fps, duration); 
    animateColor(duration, element, currentColorArray, targetColorArray, increment, stop, fpsInterval, now, then, elapsed, startTime); 
} 
function animateColor(duration, element, currentColorArray, targetColorArray, increment, stop, fpsInterval, now, then, elapsed, startTime) { 
    var step = function() { 
     if (stop) { 
      return; 
     }  
     // request another frame 
     requestAnimationFrame(function() //arguments can passed on the callback by an anonymous funtion 
     { 
      animateColor(duration, element, currentColorArray, targetColorArray, increment, stop, fpsInterval, now, then, elapsed, startTime); 
      colorTransition(element, currentColorArray, targetColorArray, increment); 
     });  
     // calc elapsed time since last loop 
     now = Date.now(); 
     elapsed = now - then;  
     // if enough time has elapsed, draw the next frame 
     if (elapsed > fpsInterval) { 
      // Get ready for next frame by setting then=now, but... 
      // Also, adjust for fpsInterval not being multiple of 16.67 
      then = now - (elapsed % fpsInterval); 
      // draw stuff here  
      var sinceStart = now - startTime;    
     } 
     if (sinceStart/1000 * 100 >= duration * 100) 
     { 
      stop = true; 
     } 
    } 
    step(); 
} 
function colorTransition(element, currentColorArray, targetColorArray, increment) { 

    // checking R 
    if (currentColorArray[0] > targetColorArray[0]) { 
     currentColorArray[0] -= increment[0]; 
     if (currentColorArray[0] <= targetColorArray[0]) { 
      increment[0] = 0; 
     } 
    } else { 
     currentColorArray[0] += increment[0]; 
     if (currentColorArray[0] >= targetColorArray[0]) { 
      increment[0] = 0; 
     } 
    }  
    // checking G 
    if (currentColorArray[1] > targetColorArray[1]) { 
     currentColorArray[1] -= increment[1]; 
     if (currentColorArray[1] <= targetColorArray[1]) { 
      increment[1] = 0; 
     } 
    } else { 
     currentColorArray[1] += increment[1]; 
     if (currentColorArray[1] >= targetColorArray[1]) { 
      increment[1] = 0; 
     } 
    }  
    // checking B 
    if (currentColorArray[2] > targetColorArray[2]) { 
     currentColorArray[2] -= increment[2]; 
     if (currentColorArray[2] <= targetColorArray[2]) { 
      increment[2] = 0; 
     } 
    } else { 
     currentColorArray[2] += increment[2]; 
     if (currentColorArray[2] >= targetColorArray[2]) { 
      increment[2] = 0; 
     } 
    }  
    // apply the new modified color 
    element.style.backgroundColor = rgb2hex(currentColorArray);  

} 
function getElementBG(elmBGColor) { 
    var bg = elmBGColor; // i.e: RGB(255, 0, 0) 
     bg = bg.match(/\((.*)\)/)[1]; 
     bg = bg.split(","); 
    for (var i = 0; i < bg.length; i++) { 
     bg[i] = parseInt(bg[i], 10); 
    } 
    if (bg.length > 3) { bg.pop(); } 
    return bg; // return array 
} 
function calculateDistance(colorArray1, colorArray2) { 
    var distance = []; 
    for (var i = 0; i < colorArray1.length; i++) { 
     distance.push(Math.abs(colorArray1[i] - colorArray2[i])); 
    } 
    return distance; 
} 
function calculateIncrement(distanceArray, fps, duration) { 
    var increment = []; 
    for (var i = 0; i < distanceArray.length; i++) { 
     increment.push(Math.abs(Math.floor(distanceArray[i]/(fps * duration)))); 
     if (increment[i] == 0) { 
      increment[i]++; 
     } 
    } 
    return increment; 
} 
function rgb2hex(colorArray) { 
    var hex = []; 
    for (var i = 0; i < colorArray.length; i++) { 
     hex.push(colorArray[i].toString(16)); 
     if (hex[i].length < 2) { hex[i] = "0" + hex[i]; } 
    } 
    return "#" + hex.join(""); 
} 
//random rgb values in array, very nice 
function generateRGB(min, max) { 
    var min = min || 0; 
    var max = max || 255; 
    var color = []; 
    for (var i = 0; i < 3; i++) { 
     var num = Math.floor(Math.random() * max); 
     while (num < min) { 
      num = Math.floor(Math.random() * max); 
     } 
     color.push(num); 
    } 
    return color; 
} 
0

使用hsl()

例如在ReactJS(這裏用的CoffeeScript)作用於SVG的文本元素,但同樣的技術將與HTML p /跨度/ H1等(顏色,而不是填充屬性)工作:

render: -> 
       #[... svg setup] 
    text 
     x: 50 
     y: 50 
     fontSize: 20 
     fill: "hsl(#{@state.el_hue}, #{@state.el_sat}%, @state.el_lum)%" 
     "Hello from randomly changing color text." 

getInitialState: -> 
    el_hue: 0 
    el_sat: 0 
    el_lum: 0 

componentDidMount: -> 
    setInterval => 
     @setState 
      el_hue: Math.random() * 360 
      el_sat: Math.random() * 100 
      el_lum: Math.random() * 100 
    , 30 

     # should do crazy things, change color etc 

在這裏,我只是做了每個時間間隔,隨機的東西與HSL丘壑,但你可以做這樣的任何事情。因此,您可以通過適當更改色調值來設置紅色到藍色的過渡。

0

下面是使用純JavaScript色彩過渡/動畫的另一種實現方式和CSS

const randomColor =() => '#' + Math.random().toString(16).substr(-6) 
 
const changeColor =() => document.body.style.backgroundColor = randomColor() 
 

 
setInterval(() => { 
 
    changeColor() 
 
}, 5000) 
 

 
// start color animation as soon as document is ready 
 
document.onreadystatechange =() => { 
 
    if (document.readyState === 'complete') { 
 
    changeColor() 
 
    } 
 
}
body { 
 
    transition: background 5s; 
 
}