::: 델파이 Tip&Trick :::

델파이 Tip&Trick 성격에 맞지 않는 광고,비방,질문의 글은 즉시 삭제하며
내용을 복사하여 사용할 경우 반드시 이곳(http://www.howto.pe.kr)을 출처로 명시하여 주세요


Category

  김영대(2003-03-07 21:49:44, Hit : 4177, Vote : 996
 달력에 그림이나 기호 그리기

안녕하세요  김영대입니다
아래 예제는 제가 전에 만들었던 스케쥴관리의 일부를 발췌한건대
폼위에는 크기가 700*500 정도의 비어있는 TImage 하고
년월을 입력받는 TMaskEdit 와 TButton 하나를 올려놓은 다음
아래 소스를 추가하세요
아래 소스는 TImage에 직접 그래픽으로 달력을 그리므로 그림같은것은
직접 Canvas 에 그리시면 됩니다

unit Unit1;

interface

uses
  Windows, Messages, SysUtils, Classes, Graphics, Controls, Forms, Dialogs,
  StdCtrls, ExtCtrls, Mask, ComCtrls, Printers;

type
  TMainForm = class(TForm)
    Button1: TButton;
    ME_yyyymm: TMaskEdit;
    Image_Schedule: TImage;
    procedure Button1Click(Sender: TObject);
    procedure FormCreate(Sender: TObject);
  private
    { Private declarations }
  public
    { Public declarations }
    procedure DrawBackGround(yyyymm: String);
    procedure DrawProgressBar(yyyymm, StartDate, EndDate: String; iColor: TColor);
    procedure DrawEvent(yyyymm, EventDate, Event: String; iColor: TColor);
end;

const
  DayOfWeekNames: array[0..6] of String =('일','월','화','수','목','금','토');
  NameHeight = 22;

var
  MainForm: TMainForm;
  OccupationEvent: array[0..31] of Integer;
  OccupationBar: array[0..31] of Integer;
  MasterCanvas: TCanvas;

implementation
{$R *.DFM}

procedure TMainForm.FormCreate(Sender: TObject);
begin
  Image_Schedule.Picture.Bitmap := TBitmap.Create; {bitmap 생성}
  MasterCanvas := Image_Schedule.Picture.Bitmap.Canvas;
  ME_yyyymm.EditMask := '####.##;0;_';
  ME_yyyymm.Text := '199905';
end;

{===============================================================================
                주어진 년월의 마지막 일자를 구한다
===============================================================================}
function Get_LastDay(yyyy,mm: Integer): Integer;
var
  MyDate: TDateTime;
  Convert_OK: Boolean;
  dd: Integer;
begin
  Convert_OK := True;
  dd := 28;
  while Convert_OK do
  begin
    try
      MyDate := EncodeDate(yyyy, mm, dd);
    except
      on EConvertError do
         Convert_OK := False;
    end;
    if Convert_OK then
      Inc(dd)
    else
      Dec(dd);
  end;
  Get_LastDay := dd;
end;

{===============================================================================
              주어진 달의 주수를 구한다(마지막 주의 의미)
===============================================================================}
function Get_weeks(yyyy,mm: Integer): Integer;
var
  MyDate: TDateTime;
  i, weeks: Integer;
  Convert_OK: Boolean;
begin
  weeks := 0;
  for i := 1 to 31 do
  begin
    try
      Convert_OK := True;
      MyDate := EncodeDate(yyyy, mm, i);
    except
      on EConvertError do
         Convert_OK := False;
    end;
    if Convert_OK then
      if (DayOfWeek(MyDate) = 7) or (Get_LastDay(yyyy,mm) = i) then {토요일이나 달의 마지막이면 1주}
        Inc(weeks);
  end;
  Get_weeks := weeks;
end;

{===============================================================================
주어진 달의 주에 해당하는 날짜from, to, 주의 첫번째 날의 요일번호를 구한다
===============================================================================}
procedure Get_weekfrto(yyyy,mm,week:Integer; var weekfrom:Integer; var weekto:Integer; var dayweek:Integer);
var
  MyDate: TDateTime;
  i, j, weeks: Integer;
  Convert_OK: Boolean;
begin
  weekfrom := 0;
  weekto   := 0;
  dayweek  := 0;

  weeks := 0;
  for i := 1 to 31 do
  begin
    try
      Convert_OK := True;
      MyDate := EncodeDate(yyyy, mm, i);
    except
      on EConvertError do
         Convert_OK := False;
    end;
    if Convert_OK then
    begin
      if (DayOfWeek(MyDate) = 7) or (Get_LastDay(yyyy,mm) = i) then {토요일이나 달의 마지막이면 1주}
        Inc(weeks);
      if weeks = week then
      begin
        weekto := i; {주의 마지막 일자}
        for j := i downto 1 do
        begin
          MyDate := EncodeDate(yyyy, mm, j);
          if (DayOfWeek(MyDate) = 1) or (j = 1) then {일요일이나 그달의 1일이면 주의 시작 일자}
          begin
            weekfrom := j; {주의 시작 일자}
            dayweek  := DayOfWeek(MyDate); {주의 첫번째 날의 요일번호}
            Break;
          end;
        end;
        System.Exit;
      end;
    end;
  end;
end;

function indc_date(yyyymmdd: String; disc: Integer): String;
var
  yyyy,mm,dd: Integer;
  MyDate: TDateTime;
  Convert_OK: Boolean;
begin
  yyyy := StrToIntDef(Copy(yyyymmdd,1,4),-1);
  mm   := StrToIntDef(Copy(yyyymmdd,5,2),-1);
  dd   := StrToIntDef(Copy(yyyymmdd,7,2),-1);
  if (yyyy <= 0) or (mm <= 0) or (dd <= 0) then
  begin
    indc_date := '';
    System.exit;
  end;

  if disc = 0 then
  begin
    try
      Convert_OK := True;
      MyDate := EncodeDate(yyyy, mm, dd);
    except
      on EConvertError do
         Convert_OK := False;
    end;
    if Convert_OK then
    begin
      indc_date := FormatDateTime('yyyymmdd', MyDate);
    end
    else
    begin
      indc_date := '';
    end;
  end
  else
  begin
    Convert_OK := True;
    try
      MyDate := EncodeDate(yyyy, mm, dd);
    except
      on EConvertError do
         Convert_OK := False;
    end;
    if Convert_OK then
    begin
      indc_date := FormatDateTime('yyyymmdd', MyDate+disc);
    end
    else
    begin
      indc_date := '';
    end;
  end;
end;

{===============================================================================
                주어진 년,월,일이 들어가는 주를 구한다
===============================================================================}
function Get_thisweek(yyyy,mm,dd: Integer): Integer;
var
  MyDate: TDateTime;
  i, weeks: Integer;
  Convert_OK: Boolean;
begin
  weeks := 0;
  for i := 1 to 31 do
  begin
    try
      Convert_OK := True;
      MyDate := EncodeDate(yyyy, mm, i);
    except
      on EConvertError do
         Convert_OK := False;
    end;
    if Convert_OK then
      if (DayOfWeek(MyDate) = 7) or (Get_LastDay(yyyy,mm) = i) then {토요일이나 달의 마지막이면 1주}
      begin
        Inc(weeks);
        if dd <= i then // 해당 주를 찾았다
          Break;
      end;
  end;
  Get_thisweek := weeks;
end;

function DayOfWeekNum(yyyymmdd: String): Integer;
var
  yyyy,mm,dd: Integer;
  MyDate: TDateTime;
begin
  yyyy := StrToIntDef(Copy(yyyymmdd,1,4),-1);
  mm   := StrToIntDef(Copy(yyyymmdd,5,2),-1);
  dd   := StrToIntDef(Copy(yyyymmdd,7,2),-1);

  Result := -1;
  try
    MyDate := EncodeDate(yyyy, mm, dd);
    Result := DayOfWeek(MyDate);
  except
    on EConvertError do
       Result := -1;
  end;
end;

procedure TMainForm.DrawBackGround(yyyymm: String);
var
  max_day, max_week: Integer;
  day_width, day_height: Integer;
  weekfrom, weekto, dayweek: Integer;
  i, j, k, yyyy, mm: Integer;
  temp: String;
begin
  yyyy := StrToIntDef(Copy(yyyymm,1,4),0);
  mm   := StrToIntDef(Copy(yyyymm,5,2),0);
  if (yyyy = 0) or (mm = 0) then
    System.Exit;

  for i := Low(OccupationEvent) to High(OccupationEvent) do
    OccupationEvent[i] := 0;
  for i := Low(OccupationBar) to High(OccupationBar) do
    OccupationBar[i] := 0;

  // 달의 마지막일자
  max_day := Get_LastDay(yyyy, mm);

  // 달의 주수
  max_week := Get_weeks(yyyy, mm);

  day_width  := Image_Schedule.Width div 7; // 가로 7칸의 각 폭
  day_height := (Image_Schedule.Height-NameHeight) div 6; // 세로 6칸의 각 높이

  MasterCanvas.Font := Self.Font;

  // 이전의 화면을 지운다
  Image_Schedule.Picture.Bitmap.Width  := Image_Schedule.Width;
  Image_Schedule.Picture.Bitmap.Height := Image_Schedule.Height;
  MasterCanvas.Brush.Color := clBtnFace;
  MasterCanvas.FillRect(Rect(0, 0, Image_Schedule.Width, Image_Schedule.Height));

  // 각 일자명을 그린다
  MasterCanvas.Brush.Color := clYellow;
  for i := 0 to 6 do
  begin
    MasterCanvas.Rectangle(day_width*i, 0,
                           day_width*(i+1), NameHeight);

    if i = 0 then // 일요일
      MasterCanvas.Font.Color := clRed
    else if i = 6 then // 토요일
      MasterCanvas.Font.Color := clBlue
    else // 평일
      MasterCanvas.Font.Color := clBlack;
    MasterCanvas.TextOut(day_width*i+5, 5, DayOfWeekNames[i]);
  end;

  // 스케줄의 각 일자의 직사각형을 그린다
  MasterCanvas.Brush.Color := clWhite;
  for i := 0 to (max_week-1) do
  begin
    for j := 0 to 6 do
      MasterCanvas.Rectangle(day_width*j, day_height*i+NameHeight,
                             day_width*(j+1), day_height*(i+1)+NameHeight);

  end;

  // 스케줄의 각 일자를 출력
  k := 0;
  for i := 0 to (max_week-1) do // 달의 주수만큼
  begin
    // 주어진 달의 주에 해당하는 날짜from, to, 주의 첫번째 날의 요일번호
    Get_weekfrto(yyyy,mm,i+1, weekfrom, weekto, dayweek);
    for j := 0 to 6 do
    begin
      if ((j+1) >= dayweek) and
         ((k+1) <= max_day) then
      begin
        Inc(k);

        if j = 0 then // 일요일
          MasterCanvas.Font.Color := clRed
        else if j = 6 then // 토요일
          MasterCanvas.Font.Color := clBlue
        else // 평일
          MasterCanvas.Font.Color := clBlack;

        MasterCanvas.Brush.Color := clWhite;
        MasterCanvas.TextOut(day_width*j+5, day_height*i+5+NameHeight, inttostr(k));
      end;
    end;
  end;
end;

procedure TMainForm.DrawProgressBar(yyyymm, StartDate, EndDate: String; iColor: TColor);
var
  day_width, day_height: Integer;
  daynum, dayweek: Integer;
  yyyy, mm: Integer;
  CurDate: String;
  iPos: Integer;
begin
  yyyy := StrToIntDef(Copy(yyyymm,1,4),0);
  mm   := StrToIntDef(Copy(yyyymm,5,2),0);

  if (yyyy = 0) or (mm = 0) then
    System.Exit;

  iPos := 1;
  CurDate := StartDate; // 시작일자
  while CurDate <= EndDate do
  begin
    if Copy(CurDate,1,6) = yyyymm then // 기준월의 날짜만 그린다
    begin
      // 요일에 출력되는 막대 카운트
      OccupationBar[StrToIntDef(Copy(CurDate,7,2),0)] :=
          OccupationBar[StrToIntDef(Copy(CurDate,7,2),0)] + 1;
      if iPos < OccupationBar[StrToIntDef(Copy(CurDate,7,2),0)] then // 해당기간의 최대 카운트
        iPos := OccupationBar[StrToIntDef(Copy(CurDate,7,2),0)];
    end;
    CurDate := indc_date(CurDate, 1); // 날짜증가
  end;

  day_width  := Image_Schedule.Width div 7; // 가로 7칸의 각 폭
  day_height := (Image_Schedule.Height-NameHeight) div 6; // 세로 6칸의 각 높이

  CurDate := StartDate; // 시작일자
  while CurDate <= EndDate do
  begin
    if Copy(CurDate,1,6) = yyyymm then // 기준월의 날짜만 그린다
    begin
      daynum := Get_thisweek(StrToInt(Copy(CurDate,1,4)), // 해당 년월일의 주번호
                             StrToInt(Copy(CurDate,5,2)),
                             StrToInt(Copy(CurDate,7,2)));

      dayweek := DayOfWeekNum(CurDate); // 요일번호

      // 막대기
      MasterCanvas.Brush.Color := iColor;
      MasterCanvas.FillRect(Rect(day_width*(dayweek-1)+ 4,
                                 day_height*(daynum-1)+ day_height-(iPos*5)-2+NameHeight,
                                 day_width*(dayweek-1)+ day_width-4,
                                 day_height*(daynum-1)+ day_height-(iPos*5)+1+NameHeight));

      if CurDate = StartDate then // 시작점 그리기
        MasterCanvas.FillRect(Rect(day_width*(dayweek-1)+ 4,
                                   day_height*(daynum-1)+ day_height-(iPos*5)-2+NameHeight-1,
                                   day_width*(dayweek-1)+ 6,
                                   day_height*(daynum-1)+ day_height-(iPos*5)-2+NameHeight+1));

      if CurDate = EndDate then // 종료점 그리기
        MasterCanvas.FillRect(Rect(day_width*(dayweek-1)+ day_width-6,
                                   day_height*(daynum-1)+ day_height-(iPos*5)+1+NameHeight-1,
                                   day_width*(dayweek-1)+ day_width-4,
                                   day_height*(daynum-1)+ day_height-(iPos*5)+1+NameHeight+1));

    end;
    CurDate := indc_date(CurDate, 1); // 날짜증가
  end;
end;

procedure TMainForm.DrawEvent(yyyymm, EventDate, Event: String; iColor: TColor);
var
  day_width, day_height: Integer;
  daynum, dayweek: Integer;
  iPos: Integer;
begin
  if Copy(EventDate,1,6) <> yyyymm then // 기준월의 날짜에 속하지 않으면...
    System.Exit;

  // 요일에 출력되는 이벤트 카운트
  OccupationEvent[StrToIntDef(Copy(EventDate,7,2),0)] :=
      OccupationEvent[StrToIntDef(Copy(EventDate,7,2),0)] + 1;
  iPos := OccupationEvent[StrToIntDef(Copy(EventDate,7,2),0)];

  day_width  := Image_Schedule.Width div 7; // 가로 7칸의 각 폭
  day_height := (Image_Schedule.Height-NameHeight) div 6; // 세로 6칸의 각 높이

  daynum := Get_thisweek(StrToInt(Copy(EventDate,1,4)), // 해당 년월일의 주번호
                         StrToInt(Copy(EventDate,5,2)),
                         StrToInt(Copy(EventDate,7,2)));

  dayweek := DayOfWeekNum(EventDate); // 요일번호

  MasterCanvas.Font.Color := iColor;
  MasterCanvas.Brush.Color := clWhite;
  MasterCanvas.TextOut(day_width*(dayweek-1)+ (day_width div 2) - (MasterCanvas.TextWidth(Event) div 2),
                       day_height*(daynum-1)+ (iPos*13) +3+ NameHeight, Event);
end;

procedure TMainForm.Button1Click(Sender: TObject);
begin
  // 기준년월의 달력을 그린다
  DrawBackGround(ME_yyyymm.Text);

  // 기준년월의 각 스케쥴과 이벤트를 그린다
  DrawProgressBar(ME_yyyymm.Text, '19990506', '19990510', clRed);
  DrawEvent(ME_yyyymm.Text, '19990507', '검토', clGreen);
  DrawEvent(ME_yyyymm.Text, '19990503', '술먹는날', clGreen);
  DrawEvent(ME_yyyymm.Text, '19990513', '술먹는날', clGreen);
  DrawEvent(ME_yyyymm.Text, '19990523', '술먹는날', clGreen);

  DrawProgressBar(ME_yyyymm.Text, '19990501', '19990505', clRed);
  DrawEvent(ME_yyyymm.Text, '19990502', '생일이브', clRed);
  DrawEvent(ME_yyyymm.Text, '19990503', '생일', clRed);

  DrawProgressBar(ME_yyyymm.Text, '19990520', '19990530', clBlue);
  DrawEvent(ME_yyyymm.Text, '19990522', '검수일', clBlue);
  DrawEvent(ME_yyyymm.Text, '19990529', '토요근무', clBlue);

  DrawProgressBar(ME_yyyymm.Text, '19990503', '19990506', clBlack);
  DrawEvent(ME_yyyymm.Text, '19990503', '시연회', clBlack);

  DrawProgressBar(ME_yyyymm.Text, '19990726', '19990726', clBlue);
  DrawEvent(ME_yyyymm.Text, '19990726', '생일', clBlue);
end;

end.





651   [일반/컴포넌트] ListBox, ComboBox에 근접검색 하기  김영대 2003/03/07 5218 1560
650   [일반/컴포넌트] StringGrid의 중간에 제목 Cell 두기  김영대 2003/03/07 5256 1328
649   [데이터베이스] 몇가지 BDE 환경정보 구하기  김영대 2003/03/07 4180 1110
648   [일반/컴포넌트] 수직 스크롤이 맨마지막으로 이동했는지 검사  김영대 2003/03/07 3772 1103
647   [데이터베이스] Query.RecordCount = -1  김영대 2003/03/07 4558 1118
646   [일반/컴포넌트] Bitmap(BMP) 크기를 변경(Stretch)하여 저장하기  김영대 2003/03/07 4719 1235
645   [일반/컴포넌트] 모든 드라이브,디렉토리 검색하여 파일 찾기  김영대 2003/03/07 4480 1120
644   [네트웍/인터넷] NT 네트워크 환경의 컴퓨터 리스트 구하기  김영대 2003/03/07 6781 1694
643   [윈도우즈 API] 윈도우즈에 프로그램으로 폰트 등록하기  김영대 2003/03/07 5238 1430
642   [일반/컴포넌트] TEdit 입력이 최대 길이를 벋어나면 focus 이동  김영대 2003/03/07 4217 1078
641   [데이터베이스] dBase 테이블의 삭제된 레코드 보기  김영대 2003/03/07 3913 1058
640   [윈도우즈 API] 윈도우즈 "시작" 메뉴의 그림 바꾸기  김영대 2003/03/07 4186 1155
639   [윈도우즈 API] 원도우 이동시 다시 그리기 설정/해제  김영대 2003/03/07 4319 1164
638   [윈도우즈 API] Windows error beep 켜기/끄기  김영대 2003/03/07 4413 1260
637   [일반/컴포넌트] 투명한(Transparent) TPanel 콤포넌트  김영대 2003/03/07 4812 1130
636   [데이터베이스] 테이블,인덱스 복사하기  김영대 2003/03/07 3846 1080
635   [일반/컴포넌트] TRichEdit 에 윗첨자, 아랫첨자 만들기  김영대 2003/03/07 4283 1118
634   [윈도우즈 API] 외부 프로그램의 크기를 내 프로그램에서 변경하기  김영대 2003/03/07 4373 1154
633   [윈도우즈 API] Ctrl-Esc, Ctrl-Alt-Del, Alt-Tab 입력 막기  김영대 2003/03/07 5403 1222
  [일반/컴포넌트] 달력에 그림이나 기호 그리기  김영대 2003/03/07 4177 996
631   [윈도우즈 API] 파일의 이동,복사,삭제 API  김영대 2003/03/07 6313 1527
630   [일반/컴포넌트] TPanel, TImage 에 Gradient 효과 주기  김영대 2003/03/07 4328 1261
629   [시스템] System, User, GDI Resource  김영대 2003/03/07 4553 1331
628   [시스템] PC에 설치된 드라이브 목록 구하기  김영대 2003/03/07 4510 1108
627   [윈도우즈 API] PrintScreen(PrtSc)키의 동작 막기  김영대 2003/03/07 5683 1276
626   [시스템] 플로피(floppy) 디스크 드라이브의 타입 구하기  김영대 2003/03/07 5811 1526
625   [일반/컴포넌트] 문자열 프로시저명으로 실제 프로시저 호출하기  김영대 2003/03/07 4836 1533
624   [일반/컴포넌트] StringGrid 수직 스크롤바의 크기를 임의로 바꾸기  김영대 2003/03/07 4225 1061
623   [일반/컴포넌트] JPEG, Bitmap 상호 변환하여 파일로 저장하기  김영대 2003/03/07 4165 1206
622   [일반/컴포넌트] Access Violation at address ?????? 로부터 소스위치 알기  김영대 2003/03/07 6266 1278
621   [일반/컴포넌트] 각도에 따라 회전한 화살표 그리기  김영대 2003/03/07 4208 1081
620   [윈도우즈 API] 작업표시줄의 "항상 위" 검사하여 설정하기  김영대 2003/03/07 4582 1177
619   [윈도우즈 API] 윈도우즈 탐색기에서 "Copy" 될 파일목록 구하기  김영대 2003/03/07 4842 1284
618   [일반/컴포넌트] TStringGrid의 마우스가 있는 Cell의 Hint 표시  김영대 2003/03/07 6387 1666
617   [윈도우즈 API] 윈도우즈 "시작" 메뉴 표시하기/감추기  김영대 2003/03/07 4335 1153
616   [일반/컴포넌트] ListBox의 Item을 버튼으로 계속 삭제하기  김영대 2003/03/07 3918 1082
615   [윈도우즈 API] 윈도우즈의 주소록(WAB) 프로그램 띄우기  김영대 2003/03/07 5301 1712
614   [윈도우즈 API] 다른 프로그램의 구동 디렉토리 알아내기  김영대 2003/03/07 4754 1285
613   [윈도우즈 API] 윈도우즈 전화걸기 화면 띄우기  김영대 2003/03/07 4460 1195
612   [일반/컴포넌트] RichEdit에서 HTML 태그를 다른색으로 표시하기  김영대 2003/03/07 5315 1190

[1][2][3][4][5][6][7][8] 9 [10]..[25] [다음 10개]
 

Copyright 1999-2019 Zeroboard / skin by zero