VC Resource 관련

리소스 ID충돌을 피하기 위하여 다음과 같은 루틴을 사용하게 된다.

#ifdef APSTUDIO_INVOKED
#ifndef APSTUDIO_READONLY_SYMBOLS
#define _APS_NEXT_RESOURCE_VALUE        263
#define _APS_NEXT_COMMAND_VALUE         33249
#define _APS_NEXT_CONTROL_VALUE         1152
#define _APS_NEXT_SYMED_VALUE           104
#endif
#endif

각각의 심볼의 의미는 다음과 같다.

_APS_NEXT_RESOURCE_VALUE is the next symbol value that will be used for a dialog resource, menu resource, and so on. The valid range for resource symbol values is 1 to 0x6FFF.


_APS_NEXT_COMMAND_VALUE is the next symbol value that will be used for a command identification. The valid range for command symbol values is 0x8000 to 0xDFFF.


_APS_NEXT_CONTROL_VALUE is the next symbol value that will be used for a dialog control. The valid range for dialog control symbol values is 8 to 0xDFFF.


_APS_NEXT_SYMED_VALUE is the next symbol value that will be issued when you manually assign a symbol value using the New command in the Symbol Browser.

여러개의 rc파일을 사용시 새로운 rc파일에서 리소스 추가 전에 충돌하지 않는 새로운 시작 ID를 할당한다. 너무 큰 숫자를 할당하게 되면 더이상 사용할 수 있는 리소스가 없어지므로 주의한다.

할당예)
#define _APS_NEXT_RESOURCE_VALUE  2000
#define _APS_NEXT_COMMAND_VALUE   42000
#define _APS_NEXT_CONTROL_VALUE   2000
#define _APS_NEXT_SYMED_VALUE     2000

Hooking과 VirtualAlloc을 통한 다른 프로그램의 ListView내용얻어오기

일반적으로 서로 다른 프로그램에서 Message전송만을 통한 Cotrol의 값의 복사가 가능하다. 하지만, 기본컨트롤이 아닌 다른 Cotrol에서 내용을 얻어오고자 할때는 Message와 VirtualAlloc을 이용해야 한다.

기본적인 순서는 다음과 같다.

1. 윈도우 핸들을 얻는다.
2. 차일드 윈도우목록에서 ListView의 핸들을 얻는다.
3. 메세지를 통해서 Item의 숫자를 얻는다.
4. ListView의 프로세스 id를 얻는다.
5. ListView의 프로세스에서 메모리를 할당한다.
6. 할당된 메모리와 Message를 이용하여 Item의 내용을 읽어온다.
7. 할당된 메모리를 해제

 다음의 예제는 어느 회사의 사용자를 검색하여 List에 출력하여 주는 간단한 델파일 프로그램을 후킹하여 사용자정보를 CSV파일로 저장하는 루틴이다.
리스트에 표현되는 정보는 순번,이름,직책,부서명,회사명,보직/담당업무,전화번호,핸드폰,이메일 이며 EnumChildWindows는 CallBack함수를 필요로한다.

HWND hFlatEdit;
HWND hListView;
CString filename;
int lineCount = 1;

BOOL CALLBACK EnumChildProc(HWND hwnd, LPARAM lPara){


 HWND hEdit = ::FindWindowEx(hwnd,NULL, “TFlatEdit”,””);
 HWND hList = ::FindWindowEx(hwnd,NULL, “TListView”, “”);
 if(hList > 0){
  hListView = hList;
  //TRACE1(“List Handle : %d\n”, hList);
 }


 if(hEdit > 0){
  //TRACE1(“Edit Handle : %d\n”,hEdit);
  hFlatEdit = hEdit;  
 }
 return TRUE;
}

void Save()
{
 HWND hwnd = ::FindWindow(NULL,”사용자검색”);
 CWnd* cwnd = CWnd::FromHandle(hwnd);   
 HINSTANCE hInstance = (HINSTANCE)(::GetWindowLong(hwnd, GWL_HINSTANCE));
 

 if(cwnd == NULL || hInstance==0){
    MessageBox(“사용자 검색창을 열어주세요”,”사용자검색창”,MB_OK);
 } else {
 
  cwnd->SetForegroundWindow();
  ::EnumChildWindows(cwnd->GetSafeHwnd(), EnumChildProc,LPARAM(cwnd));
 
  int count=(int)::SendMessage(hListView, LVM_GETITEMCOUNT, 0, 0);
  int i;
  LVITEM lvi, *_lvi;
  char item[512], subitem1[512],subitem2[512],subitem3[512],subitem4[512],subitem5[512],subitem6[512],subitem7[512],subitem8[512];
  char *_item, *_subitem1,*_subitem2,*_subitem3,*_subitem4,*_subitem5,*_subitem6,*_subitem7,*_subitem8;
  unsigned long pid;
  HANDLE process;


  GetWindowThreadProcessId(hListView, &pid);
  process=OpenProcess(PROCESS_VM_OPERATION|PROCESS_VM_READ|
   PROCESS_VM_WRITE|PROCESS_QUERY_INFORMATION, FALSE, pid);


  _lvi=(LVITEM*)VirtualAllocEx(process, NULL, sizeof(LVITEM),
   MEM_COMMIT, PAGE_READWRITE);
  _item=(char*)VirtualAllocEx(process, NULL, 512, MEM_COMMIT,
   PAGE_READWRITE);
 
  _subitem1=(char*)VirtualAllocEx(process, NULL, 512, MEM_COMMIT,
   PAGE_READWRITE);
  _subitem2=(char*)VirtualAllocEx(process, NULL, 512, MEM_COMMIT,
   PAGE_READWRITE);
  _subitem3=(char*)VirtualAllocEx(process, NULL, 512, MEM_COMMIT,
   PAGE_READWRITE);
  _subitem4=(char*)VirtualAllocEx(process, NULL, 512, MEM_COMMIT,
   PAGE_READWRITE);
  _subitem5=(char*)VirtualAllocEx(process, NULL, 512, MEM_COMMIT,
   PAGE_READWRITE);
  _subitem6=(char*)VirtualAllocEx(process, NULL, 512, MEM_COMMIT,
   PAGE_READWRITE);
  _subitem7=(char*)VirtualAllocEx(process, NULL, 512, MEM_COMMIT,
   PAGE_READWRITE);
  _subitem8=(char*)VirtualAllocEx(process, NULL, 512, MEM_COMMIT,
   PAGE_READWRITE);


  lvi.cchTextMax=512;


   
  CFileDialog fileOpen(FALSE, “CSV저장”,NULL,OFN_HIDEREADONLY,”CSV파일 (*.csv)|*.csv|”,cwnd);
  if(filename.GetLength() > 0 || fileOpen.DoModal() == IDOK){


   CString mode = “”;
   bool header = false;
   if(filename.GetLength() == 0){
    mode =”wt”;
    filename = fileOpen.GetPathName();
    header = true;
   } else {
    mode =”a”;
   }


   FILE * out = fopen(filename,mode);
   if(header == true){
    fprintf(out, “\”순번\”,\”이름\”,\”직책\”,\”부서명\”,\”회사명\”,\”보직/담당업무\”,\”전화번호\”,\”핸드폰\”,\”이메일\”\n”);
   }
     
   for(i=0; i<count; i++) {
    lvi.iSubItem=0;
    lvi.pszText=_item;
    ::WriteProcessMemory(process, _lvi, &lvi, sizeof(LVITEM), NULL);
    ::SendMessage(hListView, LVM_GETITEMTEXT, (WPARAM)i, (LPARAM)_lvi);


    lvi.iSubItem=1;
    lvi.pszText=_subitem1;
    ::WriteProcessMemory(process, _lvi, &lvi, sizeof(LVITEM), NULL);
    ::SendMessage(hListView, LVM_GETITEMTEXT, (WPARAM)i, (LPARAM)_lvi);


    lvi.iSubItem=2;
    lvi.pszText=_subitem2;
    ::WriteProcessMemory(process, _lvi, &lvi, sizeof(LVITEM), NULL);
    ::SendMessage(hListView, LVM_GETITEMTEXT, (WPARAM)i, (LPARAM)_lvi);


    lvi.iSubItem=3;
    lvi.pszText=_subitem3;
    ::WriteProcessMemory(process, _lvi, &lvi, sizeof(LVITEM), NULL);
    ::SendMessage(hListView, LVM_GETITEMTEXT, (WPARAM)i, (LPARAM)_lvi);


    lvi.iSubItem=4;
    lvi.pszText=_subitem4;
    ::WriteProcessMemory(process, _lvi, &lvi, sizeof(LVITEM), NULL);
    ::SendMessage(hListView, LVM_GETITEMTEXT, (WPARAM)i, (LPARAM)_lvi);


    lvi.iSubItem=5;
    lvi.pszText=_subitem5;
    ::WriteProcessMemory(process, _lvi, &lvi, sizeof(LVITEM), NULL);
    ::SendMessage(hListView, LVM_GETITEMTEXT, (WPARAM)i, (LPARAM)_lvi);


    lvi.iSubItem=6;
    lvi.pszText=_subitem6;
    ::WriteProcessMemory(process, _lvi, &lvi, sizeof(LVITEM), NULL);
    ::SendMessage(hListView, LVM_GETITEMTEXT, (WPARAM)i, (LPARAM)_lvi);


    lvi.iSubItem=7;
    lvi.pszText=_subitem7;
    ::WriteProcessMemory(process, _lvi, &lvi, sizeof(LVITEM), NULL);
    ::SendMessage(hListView, LVM_GETITEMTEXT, (WPARAM)i, (LPARAM)_lvi);


    //lvi.iSubItem=8;
    //lvi.pszText=_subitem8;
    //::WriteProcessMemory(process, _lvi, &lvi, sizeof(LVITEM), NULL);
    //::SendMessage(hListView, LVM_GETITEMTEXT, (WPARAM)i, (LPARAM)_lvi);


    ::ReadProcessMemory(process, _item, item, 512, NULL);
    ::ReadProcessMemory(process, _subitem1, subitem1, 512, NULL);
    ::ReadProcessMemory(process, _subitem2, subitem2, 512, NULL);
    ::ReadProcessMemory(process, _subitem3, subitem3, 512, NULL);
    ::ReadProcessMemory(process, _subitem4, subitem4, 512, NULL);
    ::ReadProcessMemory(process, _subitem5, subitem5, 512, NULL);
    ::ReadProcessMemory(process, _subitem6, subitem6, 512, NULL);
    ::ReadProcessMemory(process, _subitem7, subitem7, 512, NULL);
    //::ReadProcessMemory(process, _subitem8, subitem8, 512, NULL);



    char output[2048];
    sprintf(output, “\”%d\”,\”%s\”,\”%s\”,\”%s\”,\”%s\”,\”%s\”,\”%s\”,\”%s\”,\”%s\””,lineCount,item,subitem1,subitem2,subitem3,subitem4,subitem5,subitem6,subitem7);
    fprintf(out,”%s\n”,output);
    lineCount++;
   }
   fclose(out);
  }


  ::VirtualFreeEx(process, _lvi, 0, MEM_RELEASE);
  ::VirtualFreeEx(process, _item, 0, MEM_RELEASE);
  ::VirtualFreeEx(process, _subitem1, 0, MEM_RELEASE);
  ::VirtualFreeEx(process, _subitem2, 0, MEM_RELEASE);
  ::VirtualFreeEx(process, _subitem3, 0, MEM_RELEASE);
  ::VirtualFreeEx(process, _subitem4, 0, MEM_RELEASE);
  ::VirtualFreeEx(process, _subitem5, 0, MEM_RELEASE);
  ::VirtualFreeEx(process, _subitem6, 0, MEM_RELEASE);
  ::VirtualFreeEx(process, _subitem7, 0, MEM_RELEASE);

 }
}

한개의 Table에서 2개의 Auto Timestamp 컬럼 사용하기(MySQL5.x)

다음은 MySQL5.0부터 추가된 트리거를 이용한 2개이상의 timestamp필드를 업데이트 하는 방법을 기술한다.

It’s very common to be in need of more than one auto-TIMESTAMP column in one table, firing on different operations: The blog you’re reading right now for example stores the time an article has first been published (corresponds to an INSERT operation) as well as the time an article has last been updated (corresponds to an UPDATE operation).

What you possibly want to do to achieve this is something like the following:

 CREATE TABLE blog_entries (
 `published` TIMESTAMP DEFAULT CURRENT_TIMESTAMP,
 `updated` TIMESTAMP DEFAULT ‘0000-00-00 00:00:00’ ON UPDATE CURRENT_TIMESTAMP,
 `title` VARCHAR(128)
 );

But this still won’t work in 5.0. The server returns an error:
ERROR 1293 (HY000): Incorrect table definition; there can be only one
TIMESTAMP column with CURRENT_TIMESTAMP in DEFAULT or ON UPDATE clause

Instead we have to take a different approach and skip the built-in functionality of TIMESTAMP columns completely. Let’s define the table first:

 CREATE TABLE blog_entries (
  `published` TIMESTAMP DEFAULT ‘0000-00-00 00:00:00’,
  `updated` TIMESTAMP DEFAULT ‘0000-00-00 00:00:00’,
  `title` VARCHAR(128)
 );

We now have a table with two TIMESTAMP columns that both almost act as ordinary DATETIME columns, which means they are not set automatically. Let’s add the intended functionality for the published column again using a TRIGGER that fires every time just before a new row is inserted into the table:

 CREATE TRIGGER blog_entries_publish BEFORE INSERT ON `blog_entries`
 FOR EACH ROW SET NEW.published = NOW(), NEW.updated = ‘0000-00-00 00:00:00’;

In a BEFORE TRIGGER you can access and change the value to be inserted/updated via the NEW keyword. That’s what we did with this TRIGGER: Whenever a new row will be added to the table blog_entries our TRIGGER fires and sets the value for the column published to the actual time at the moment of the INSERT and the column updated to its default value of a zero timestamp (meaning in our example that this row has never been updated).
Let’s do almost the same for the UPDATE operation, just setting updated to the current time here and keeping the OLD value (the one being stored in the table before the UPDATE) for the column published:

 CREATE TRIGGER blog_entries_update BEFORE UPDATE ON `blog_entries`
 FOR EACH ROW SET NEW.updated = NOW(), NEW.published = OLD.published;

You’ll notice however that this is a very strict version of auto-TIMESTAMP columns: Our TRIGGERs override whatever values we try to INSERT or UPDATE for the two columns. This means it’s actually no longer possible to set these two columns manually.
In some situations this might be just what you want: For auditing, for example. You can now be sure that nobody ever messes around with your TIMESTAMPs (at least nobody with no rights to DROP your TRIGGERs). But if you want to be less strict with your users you can still build something that more closely emulates the built-in behaviour of TIMESTAMP columns:

 CREATE TABLE blog_entries (
  `published` TIMESTAMP NULL DEFAULT NULL,
  `updated` TIMESTAMP NULL DEFAULT NULL,
  `title` VARCHAR(128)
 );

CREATE TRIGGER blog_entries_publish BEFORE INSERT ON `blog_entries` FOR EACH ROW   SET
  NEW.published = IFNULL(NEW.published, NOW()),
  NEW.updated = IFNULL(NEW.updated, ‘0000-00-00 00:00:00’);

CREATE TRIGGER blog_entries_update BEFORE UPDATE ON `blog_entries` FOR EACH ROW SET
  NEW.updated = IF(NEW.updated = OLD.updated OR NEW.updated IS NULL, NOW(), NEW.updated),
  NEW.published = IFNULL(NEW.published, OLD.published);

The magic functionality of our columns is now only triggered if a NULL value is written to them. By setting the two columns to DEFAULT NULL we make sure this is the case if you just leave them out in your INSERT statements.
For the UPDATE operation it’s not as easy: If you don’t provide a value for the updated column its NEW value will be the same as its OLD and not the default of NULL during execution of the TRIGGER. So you’ll have to check for this situation as well, making it impossible to explicitly set the value of the updated column to the same as it was before with the above code. But you could of course change it and define NULL to mean “same as before” for the updated column:

 CREATE TRIGGER blog_entries_update BEFORE UPDATE ON `blog_entries` FOR EACH ROW SET
  NEW.updated = CASE
    WHEN NEW.updated IS NULL THEN OLD.updated
    WHEN NEW.updated = OLD.updated THEN NOW()
   ELSE NEW.updated
 END,
  NEW.published = IFNULL(NEW.published, OLD.published);

That’s the nice thing with TRIGGERs: Just taylor it to your needs!


출처 : http://www.futhark.ch/mysql/108.html 

WGL에서 DC전환

Opengl을 이용한 MDI Application에서 Current View가 보존되지 않으면 ChildWnd의 다른 View에 업데이트 될수 있다.

OnActivateView를 이용해 Current를 자신의 dc로 유지한다.



void CModelViewerView:: OnActivateView( BOOL bActivate, CView* pActivateView, CView* pDeactiveView )
{


  CView::OnActivateView(bActivate, pActivateView, pDeactiveView);


  if(bActivate == TRUE){
    wglMakeCurrent(m_pDC->GetSafeHdc(), m_hRC);
  }


}



gl명령어를 쓰기 위한 조건에서


HDC hdc = wglGetCurrentDC();
HGLRC hrc = wglGetCurrentContext();
wglMakeCurrent(m_pDC->GetSafeHdc(), m_hRC);


 // here gl command
wglMakeCurrent(hdc,hrc);


와 같이 current dc를 보존한다.

FLTK1.1 VC60 컴파일 에러

FLTK를 vc에서 컴파일하면 다음과 같은 링크에러가 발생한다.


error LNK2005: “public: void __thiscall Fl_Widget::label(char const *)” (?label@Fl_Widget@@QAEXPBD@Z) already defined


해결책


SomeWidget->label(FL_NORMAL_LABEL,“theLabel”,);


현재로서는 vc가 obj를 중복링크를 허용하지 않으므로 다른해결책이 없어보임

MySQL 4.1 한글설정

Mysql 4.1에서부터는 전체 문자세트를 지정해도 한글이 적용이 안된다.

 전체 문자세트를 euckr로 지정한뒤  각각의 Table단위와 Connection에서 문자세트를 euckr로 지정하여 준다..


– Connection에 적용 –


jdbc:mySQL://localhost:3309/req_db?autoReconnect=true&useUnicode=true&characterENCODING=euckr&mysqlENCODING=euckr


 characterENCODING=euckr


 mysqlENCODING=euckr


– 테이블 단위의 적용 –



create table user (


   …..(필드들)…
 ) character set euckr;

VC.NET2003 빌드넘버 자동증가 매크로







프로젝트 빌드넘버 자동 증가 시키기!!

이것의 핵심은 VC의 IDE에서도 매크로를 사용할 수 있다는 것이고, IDE에서 컴파일을 수행하고 나면 그 컴파일 결과를 돌려 준다는 것입니다. 여러분도 알다시피 컴파일을 수행하고 나면 그 수행결과들이 Output/Build창에 나타나듯이…

그럼 먼저 Sample로 사용할 프로젝트를 하나 생성합니다. 저의 경우 AutoIncBuildNumber 라는 프로젝트를 생성했습니다.

이렇게 프로젝트를 생성한 후 프로젝트의 리소스들을 엽니다.

File -> Open -> AutoIncBuildNumber.rc and AutoIncBuildNumber.rc2

파일오픈(열기) 다이얼로그에서 Open as를 Auto에서 Text로 바꾸어서 열어야 합니다. 이렇게 파일을 연 후에 AutoIncBuildNumber.rc 에서 아래와 같은 내용을 찾습니다.

///////////////////////////////////////////////////////////////////////

//

// Version

//

VS_VERSION_INFO VERSIONINFO

FILEVERSION 1,0,0,1

PRODUCTVERSION 1,0,0,1

FILEFLAGSMASK 0x3fL

#ifdef _DEBUG

FILEFLAGS 0x1L

#else

FILEFLAGS 0x0L

#endif

FILEOS 0x4L

FILETYPE 0x1L

FILESUBTYPE 0x0L

BEGIN

BLOCK “StringFileInfo”

BEGIN

BLOCK “040904b0”

BEGIN

VALUE “Comments”, “Sample Application\0”

VALUE “CompanyName”, “Microsoft Corp.\0”

VALUE “FileDescription”, “MyProject MFC Application\0”

VALUE “FileVersion”, “1, 0, 0, 1\0”

VALUE “InternalName”, “MyProject\0”

VALUE “LegalCopyright”, “Copyright (C) 1999\0”

VALUE “OriginalFilename”, “MyProject.EXE\0”

VALUE “ProductName”, “MyProject Application\0”

VALUE “ProductVersion”, “1, 0, 0, 1\0”

END

END

BLOCK “VarFileInfo”

BEGIN

VALUE “Translation”, 0x409, 1200

END

END

즉, 버젼에 관계된 내용이있는 곳을 찾습니다. 그리고 이 부분을 잘라내기 한 다음 AutoIncBuildNumber.rc2 파일의

/////////////////////////////////////////////////////////////////////////////

// Add manually edited resources here…

라고 되어 있는 부분 아래로 붙여 넣습니다. 붙여 넣기 한 후에 한 줄을 추가 해 줍니다.

///////////////////////////////////////////////////////////////////////

//

// Version

//

#include “VersionNo.h”

VS_VERSION_INFO VERSIONINFO

FILEVERSION 1,0,0,1

PRODUCTVERSION 1,0,0,1

FILEFLAGSMASK 0x3fL

#ifdef _DEBUG

FILEFLAGS 0x1L

#else

FILEFLAGS 0x0L

#endif

FILEOS 0x4L

FILETYPE 0x1L

FILESUBTYPE 0x0L

BEGIN

BLOCK “StringFileInfo”

BEGIN

BLOCK “040904b0”

BEGIN

VALUE “Comments”, “Sample Application\0”

VALUE “CompanyName”, “Microsoft Corp.\0”

VALUE “FileDescription”, “MyProject MFC Application\0”

VALUE “FileVersion”, “1, 0, 0, 1\0”

VALUE “InternalName”, “MyProject\0”

VALUE “LegalCopyright”, “Copyright (C) 1999\0”

VALUE “OriginalFilename”, “MyProject.EXE\0”

VALUE “ProductName”, “MyProject Application\0”

VALUE “ProductVersion”, “1, 0, 0, 1\0”

END

END

BLOCK “VarFileInfo”

BEGIN

VALUE “Translation”, 0x409, 1200

END

END

추가 한 후에

///////////////////////////////////////////////////////////////////////

//

// Version

//

#include “VersionNo.h”

VS_VERSION_INFO VERSIONINFO

FILEVERSION FILEVER

PRODUCTVERSION PRODUCTVER

FILEFLAGSMASK 0x3fL

#ifdef _DEBUG

FILEFLAGS 0x1L

#else

FILEFLAGS 0x0L

#endif

FILEOS 0x4L

FILETYPE 0x1L

FILESUBTYPE 0x0L

BEGIN

BLOCK “StringFileInfo”

BEGIN

BLOCK “040904b0”

BEGIN

VALUE “Comments”, “Sample Application\0”

VALUE “CompanyName”, “Microsoft Corp.\0”

VALUE “FileDescription”, “MyProject MFC Application\0”

VALUE “FileVersion”, STRFILEVER

VALUE “InternalName”, “MyProject\0”

VALUE “LegalCopyright”, “Copyright (C) 1997\0”

VALUE “OriginalFilename”, “MyProject.EXE\0”

VALUE “ProductName”, “MyProject Application\0”

VALUE “ProductVersion”, STRPRODUCTVER

END

END

BLOCK “VarFileInfo”

BEGIN

VALUE “Translation”, 0x409, 1200

END

END

굴게 쓰여진 글씨처럼 FILEVERSION과 PRODUCTVERSION에 관련된 숫자들과 문자열을 바꾸어 줍니다. 이 바꾸어진 문장들을 VersionNo.h 파일에 define해 줍니다.

굵게 쓰여진 글씨처럼 #include “VersionNo.h” 라는 문장을 추가 해 주신 후 New -> C/C++ Header File 로 VersionNo.h파일을 만들어 줍니다.

#define FILEVER 1,0,0,1

#define PRODUCTVER 1,0,0,1

#define STRFILEVER “1, 0, 0, 1\0”

#define STRPRODUCTVER “1, 0, 0, 1\0”

위 문장들을 VersionNo.h에 추가 해 줍니다.

여기까지 따라 하셨다면, 다음으로 IDE에서 컴파일 종료 이벤트를 받아서 실행 할 Macro 파일을 생성해 줍니다. 이 메크로의 역할이 여러분이 상상하시듯이 컴파일 이벤트를 받은 후 프로젝트의 버젼을 증가시켜 주는 것 입니다.

New -> Macro File, AutoIncBuildNumber

파일을 만든 후 아래 내용을 넣으십시오.

Imports EnvDTE
Imports System.Diagnostics

Public Module AutoVersion

Function GetProjectDir(ByVal FullName)

Dim proj_path
proj_path = Split(StrReverse(FullName), “\”, -1, 1)

Dim count
count = UBound(proj_path)

Dim full_path
full_path = “”
Dim i

For i = 1 To count
full_path = full_path & “\” & proj_path(i)
Next

GetProjectDir = StrReverse(full_path)

End Function

Sub ReplaceText(ByVal objSel As TextSelection, _
ByVal count As Integer, ByVal incrementby As Integer, _
ByVal Type As Integer)

Dim strTemp As String
Dim i

strTemp = “”
objSel.EndOfLine()

If Type = 0 Then
For i = 1 To count
If strTemp.StartsWith(“,”) = True Then
Exit For
Else
objSel.CharLeft(True, 1)
strTemp = objSel.Text
End If
Next
strTemp = strTemp.Remove(0, 1)
strTemp = strTemp + incrementby
objSel.Text = “,” & strTemp
Else

For i = 1 To count
If strTemp.StartsWith(“,”) = True Then
Exit For
Else
objSel.CharLeft(True, 1)
strTemp = objSel.Text
End If
Next
strTemp = strTemp.Remove(strTemp.Length – 3, 3)
strTemp = strTemp.Remove(0, 1)
strTemp = strTemp + incrementby
objSel.Text = “,” & strTemp & “\” & “0” & “”””

End If

End Sub

Dim WithEvents bldevents As BuildEvents
Dim applicationObject As EnvDTE.DTE

Sub BuildDoneEvents()
Dim addInInstance As EnvDTE.AddIn

applicationObject = CType(Application, EnvDTE.DTE)
bldevents = CType(applicationObject.Events. _
BuildEvents, EnvDTE.BuildEvents)
End Sub

Private Sub bldevents_OnBuildDone(ByVal _
Scope As EnvDTE.vsBuildScope, _
ByVal Action As EnvDTE. _
vsBuildAction) Handles _
bldevents.OnBuildDone

Dim full_path
full_path = GetProjectDir(DTE.ActiveDocument.Path)

full_path = full_path & “versionno.h”

Dim doc As Document
DTE.ItemOperations.OpenFile(full_path)

Dim objDoc As TextDocument

objDoc = DTE.ActiveDocument.Object(“TextDocument”)

Dim objSel As TextSelection = _
DTE.ActiveDocument.Selection

objSel.StartOfDocument()

ReplaceText(objSel, 6, 1, 0)
objSel.LineDown()
objSel.StartOfLine()
ReplaceText(objSel, 6, 1, 0)
objSel.LineDown()
objSel.StartOfLine()
ReplaceText(objSel, 6, 1, 100)
objSel.LineDown()
objSel.StartOfLine()
ReplaceText(objSel, 6, 1, 100)
ActiveDocument.Save()
ActiveDocument.Close()
End Sub

End Module

그리고, 저장… 합니다.

머 대충 보셔서 아시겠지만, 컴파일 이벤트를 받으면 VersionNo.h파일의 버젼 정보들을 찾아서 값을 증가시켜 주는 머 그런 내용입니다.

여기까지 하셨다면 다 하신 겁니다.

마지막으로 가장 중요한, 메크로를 등록하는 과정이 남았습니다. 메크로를 등록하지 않는 다면 지금까지 한 내용은 의미가 없어지겠지요.

Tools -> Customize… 메뉴를 선택 하신 후 Add-in and Macro Files 탭에 보시면 조금전 만든 AutoIncBuildNumber라는 메크로가 보일 것 입니다. 가볍게 체크…해 주세요.

자 이제 완료… 이제 컴파일을 해 보세요. 몇번이고 하셔도 상관 없겠지요… 그리고 난 후 VersionNo.h파일을 열어 보십시오. 빌드 넘버가 이쁘게 증가 되어 있을 겁니다.

== 출처 == codeguru & devpia & Macro수정:본인

Managed C++에서 new 사용시 에러수정

메모리 사용을 추적하기 위해 MFC는 new 연산자가 MFC의 디버그 버전에 사용될 때 new 연산자를 다시 정의합니다. new 연산자를 다시 정의하므로 관리되는 클래스의 인스턴스를 MFC 응용 프로그램에서 만들면 오류가 발생할 수 있습니다. 이런 현상은 일반적으로 기존 MFC 코드를 공용 언어 런타임에 포팅할 때 발생합니다. 릴리스 빌드의 경우, MFC에서 new 연산자를 다시 정의하지 않으므로 이 오류는 발생하지 않습니다.

다음 예제에서는 .cpp 파일에 있는 관리되는 코드가 String 클래스의 인스턴스를 만듭니다. 이렇게 하면 MFC 응용 프로그램의 디버그 버전에서 컴파일할 때 C3828 컴파일러 오류가 발생합니다.

#using <mscorlib.dll>
using namespace System;

// MFC code

String* s;
s = new String(“Hello world!”);
이런 오류가 발생하지 않게 하려면 관리되는 형식의 인스턴스를 만들기 전에 #undef 및 push_macro 지시문을 사용하여 new 연산자의 정의를 임시로 해제하십시오. 관리되는 코드의 마지막 행 뒤에 pop_macro를 사용하여 new 연산자의 이전 정의를 복원하십시오.

#pragma push_macro(“new”)
#undef new

String* s;
s = new String(“Hello world!”);
#pragma pop_macro(“new”)

출처 – MSDN –