国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 學院 > 開發設計 > 正文

一個畫漸變的方法

2019-11-18 18:42:51
字體:
來源:轉載
供稿:網友
一個畫漸變的方法,希望大家能夠幫我回答這個問題:
http://www.csdn.net/expert/topic/490/490274.shtm

(*
  ———————————————原理:————————————————
  對于任何一種線性漸變(就是最常見的那種),在由起點和終點定義的漸變區
  內,像素的RGB分量對于X和Y坐標的偏導數都是常量。于是我們可以先用極小
  的代價來計算出這個二元方程的初始值,然后使用累加遞推的方法計算出所有
  的值。
  ———————————————注 1:————————————————
  漸變區:由分別經過起點和終點,并垂直于這兩點連線的平行直線和繪圖區域
  的邊界圍成的區域。在這個區域以外的像素不再有漸變。
  ———————————————注 2:————————————————
  為了簡化編程,我使用了浮點數來進行累加計算。實際上可以先用移位操作來
  “放大”顏色值以提高累加時的精度,寫入位圖時再用移位“縮小”來恢復實
  際的值。
  ———————————————測試:————————————————
  該方法在Delphi6下調試通過。在Duron800,1152 X 864下采取重回1000次取
  平均值的方法測試。
  100 X 100:平均為4ms;
  500 X 200:平均為18ms;
  (我同時還在聽MP3:P)
*)

function SSDrawGradient(ACanvas: TCanvas; AClipRect: TRect;
  FromPoint, ToPoint: TPoint; FromColor, ToColor: TColor): Boolean;
type
  TSSGradientDirection=(gdEast, gdWest, gdNorth, gdSouth, gdOther);
var
  buf:TBitmap;
  w,h,y,x,XOffset,ir,ig,ib,pw,ph:Integer;
  c1, c2: TColor;
  r1,g1,b1,r2,g2,b2,br,bg,bb,rmax,rmin,gmax,gmin,bmax,bmin: Byte;
  kx,ky,kx0,ky0,rx0,gx0,bx0,r0,g0,b0,drx,dry,dgx,dgy,dbx,dby,dr,dg,db: Double;
  P : PByteArray;

  function GetStep(V1, V2, V3:Integer): Double;
  begin
    if V2=V1 then Result:=0
    else Result:=V3/(V2-V1);
  end;

begin
  Result:=False;
  if (FromPoint.Y=ToPoint.Y)and(FromPoint.X=ToPoint.X) then Exit;
  buf:=TBitmap.Create;
  try
    //初始化緩沖區
    buf.PixelFormat:=pf24bit;
    w:=WidthOfRect(ACliPRect);
    buf.Width:=w;
    h:=HeightOfRect(AClipRect);
    buf.Height:=h;
    //為了防止運算溢出而設的檢查
    if (w>Screen.Width)or(h>Screen.Height) then Exit;

    //讀取漸變起點和終點的RGB值
    c1:=ColorToRGB(FromColor);
    c2:=ColorToRGB(ToColor);
    r1:=GetRValue(c1);
    g1:=GetGValue(c1);
    b1:=GetBValue(c1);
    r2:=GetRValue(c2);
    g2:=GetGValue(c2);
    b2:=GetBValue(c2);
    if r1>r2 then begin rmin:=r2; rmax:=r1 end
    else begin rmin:=r1; rmax:=r2 end;
    if g1>g2 then begin gmin:=g2; gmax:=g1 end
    else begin gmin:=g1; gmax:=g2 end;
    if b1>b2 then begin bmin:=b2; bmax:=b1 end
    else begin bmin:=b1; bmax:=b2 end;
    pw:=Abs(ToPoint.X-FromPoint.X);
    ph:=Abs(ToPoint.Y-FromPoint.Y);
    kx:=pw/Sqrt(ph*ph+pw*pw);
    ky:=ph/Sqrt(ph*ph+pw*pw);
   
    //計算出RGB值相對于XY軸的線性變化系數
    drx:=GetStep(AClipRect.Left, AClipRect.Right, Round((r2-r1)*kx));
    dry:=GetStep(AClipRect.Top, AClipRect.Bottom, Round((r2-r1)*ky));
    dgx:=GetStep(AClipRect.Left, AClipRect.Right, Round((g2-g1)*kx));
    dgy:=GetStep(AClipRect.Top, AClipRect.Bottom, Round((g2-g1)*ky));
    dbx:=GetStep(AClipRect.Left, AClipRect.Right, Round((b2-b1)*kx));
    dby:=GetStep(AClipRect.Top, AClipRect.Bottom, Round((b2-b1)*ky));

    //計算出矩形左上角的RGB值,備用
    kx0:=GetStep(FromPoint.X, ToPoint.X, FromPoint.X);
    ky0:=GetStep(FromPoint.Y, ToPoint.Y, FromPoint.Y);
    r0:=r1+(kx0+ky0)*r2;
    g0:=g1+(kx0+ky0)*g2;
    b0:=b1+(kx0+ky0)*b2;

    //這三個變量是每個掃描線的第一個點的RGB值
    rx0:=r0;
    gx0:=g0;
    bx0:=b0;
    for y:=0 to h-1 do
    begin
      XOffset:=0;
      //dr意思是Double類型的紅色值,其他類推
      dr:=rx0;
      dg:=gx0;
      db:=bx0;
      P := buf.ScanLine[y];
      for x:=0 to w-1 do
      begin
        //ir的意思是整型的紅色值,其他類推
        //之所以要先轉成整型,是因為我覺得整型的比較也許會比浮點快一點
        //反正都要三次Round的,不如早做……
        ir:=Round(dr);
        ig:=Round(dg);
        ib:=Round(db);
        //br的意思是字節型的紅色值
        br:=Max(Min(rmax,ir),rmin);
        bg:=Max(Min(gmax,ig),gmin);
        bb:=Max(Min(bmax,ib),bmin);
        //按照偏移量設置RGB值
        P[XOffset]:=bb;
        P[XOffset+1]:=bg;
        P[XOffset+2]:=br;

        if FromPoint.X<>ToPoint.X then
        begin
          //下一個像素的RGB值分別按照一定的系數遞增
          dr:=dr+drx;
          dg:=dg+dgx;
          db:=db+dbx;
        end;
        //因為我定義的P是字節型的數組,所以這里遞增“3”,避免使用乘法
        Inc(XOffset, 3);
      end;
      if FromPoint.Y<>ToPoint.Y then
      begin
        //按照RGB在Y軸方向上的變化規律計算下一行的第一個像素RGB值
        rx0:=rx0+dry;
        gx0:=gx0+dgy;
        bx0:=bx0+dby;
      end;
    end;
    //將緩沖區復制到目標上
    BitBlt(ACanvas.Handle, AClipRect.Left, AClipRect.Top, w, h,
      buf.Canvas.Handle, 0, 0, SRCCOPY);
    Result:=True;
  finally
    buf.Free;
  end;
end;


上一篇:讀寫CMOS內存

下一篇:掃雷程序思想講解

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
學習交流
熱門圖片

新聞熱點

疑難解答

圖片精選

網友關注

主站蜘蛛池模板: 枣庄市| 虹口区| 航空| 朝阳区| 庆阳市| 黔江区| 沙湾县| 泸水县| 芦溪县| 青铜峡市| 柘荣县| 游戏| 南城县| 普宁市| 孟连| 莒南县| 德格县| 喀喇沁旗| 高安市| 海宁市| 自贡市| 五原县| 冀州市| 新平| 德庆县| 酒泉市| 庄河市| 贡山| 保定市| 西乌| 保山市| 平乐县| 富川| 郧西县| 长阳| 新竹县| 宾川县| 三江| 修文县| 西城区| 巫山县|