mirror of
https://github.com/WinampDesktop/winamp.git
synced 2024-09-24 15:54:12 +00:00
2156 lines
64 KiB
C++
2156 lines
64 KiB
C++
|
#include "main.h"
|
||
|
#include <stdio.h>
|
||
|
#include "../nu/ns_wc.h"
|
||
|
#include "resource.h"
|
||
|
#include "../nu/listview.h"
|
||
|
#include "../nu/DialogSkinner.h"
|
||
|
#include "../nu/ChildSizer.h"
|
||
|
#include "config.h"
|
||
|
#include "../../General/gen_ml/gaystring.h"
|
||
|
#include "../Winamp/burn.h"
|
||
|
#include "../Winamp/strutil.h"
|
||
|
#include <std::vector>
|
||
|
#include "../nu/AutoChar.h"
|
||
|
#include "../nu/AutoWide.h"
|
||
|
#include <api/service/waServiceFactory.h>
|
||
|
#include "../playlist/ifc_playlistloadercallback.h"
|
||
|
#include "../playlist/api_playlistmanager.h"
|
||
|
#include <imapi.h>
|
||
|
#include <imapierror.h>
|
||
|
#include <shlwapi.h>
|
||
|
#include <strsafe.h>
|
||
|
|
||
|
//shit to finish:
|
||
|
//-erase CDRWs
|
||
|
//-cache the veritas handle
|
||
|
//-recurse add folders
|
||
|
//-check for available space on HD before burning
|
||
|
//-the resampling in convert
|
||
|
|
||
|
|
||
|
#define WM_EX_OPCOMPLETED (WM_USER + 0x100)
|
||
|
|
||
|
class PLCallBack : ifc_playlistloadercallback
|
||
|
{
|
||
|
public:
|
||
|
PLCallBack(void) : fileList(0) {};
|
||
|
~PLCallBack(void) {};
|
||
|
public:
|
||
|
void OnFile(const wchar_t *filename, const wchar_t *title, int lengthInMS, ifc_plentryinfo *info)
|
||
|
{
|
||
|
fileList->push_back(new GayString(AutoChar(filename)));
|
||
|
}
|
||
|
RECVS_DISPATCH;
|
||
|
public:
|
||
|
std::vector<GayString*> *fileList;
|
||
|
};
|
||
|
|
||
|
#define CBCLASS PLCallBack
|
||
|
START_DISPATCH;
|
||
|
VCB(IFC_PLAYLISTLOADERCALLBACK_ONFILE, OnFile)
|
||
|
END_DISPATCH;
|
||
|
#undef CBCLASS
|
||
|
|
||
|
class PLCallBackW : ifc_playlistloadercallback
|
||
|
{
|
||
|
public:
|
||
|
PLCallBackW(void) : fileList(0) {};
|
||
|
~PLCallBackW(void) {};
|
||
|
public:
|
||
|
void OnFile(const wchar_t *filename, const wchar_t *title, int lengthInMS, ifc_plentryinfo *info)
|
||
|
{
|
||
|
fileList->push_back(new GayStringW(filename));
|
||
|
}
|
||
|
RECVS_DISPATCH;
|
||
|
public:
|
||
|
std::vector<GayStringW*> *fileList;
|
||
|
};
|
||
|
|
||
|
#define CBCLASS PLCallBackW
|
||
|
START_DISPATCH;
|
||
|
VCB(IFC_PLAYLISTLOADERCALLBACK_ONFILE, OnFile)
|
||
|
END_DISPATCH;
|
||
|
#undef CBCLASS
|
||
|
|
||
|
static INT_PTR WINAPI DlgProc(HWND hwndDlg, UINT uMsg, WPARAM wParam, LPARAM lParam);
|
||
|
|
||
|
#include "../burnlib/burnlib.h"
|
||
|
|
||
|
static W_ListView m_statuslist;
|
||
|
|
||
|
static HWND m_hwndstatus;
|
||
|
static char m_cdrom;
|
||
|
static int m_is_cdrw, m_availsecs;
|
||
|
static int m_max_speed;
|
||
|
static int m_dragging, m_drag_item;
|
||
|
static HWND prevWnd = NULL;
|
||
|
itemRecordListW itemCache[100] = {0};
|
||
|
static int percentCompleted = 0;
|
||
|
static DWORD pidBurner = 0;
|
||
|
|
||
|
static HFONT hPLFont = NULL;
|
||
|
|
||
|
static int LETTERTOINDEX(char c)
|
||
|
{
|
||
|
c = (char)toupper(c);
|
||
|
if (c < 'A') c = 'A';
|
||
|
if (c > 'Z') c = 'Z';
|
||
|
return c -'A';
|
||
|
}
|
||
|
|
||
|
#include "../winamp/wa_ipc.h"
|
||
|
|
||
|
|
||
|
#define TIMER_NOTIFYINFO_ID 1985
|
||
|
#define TIMER_NOTIFYINFO_DELAY 200
|
||
|
|
||
|
|
||
|
static ChildWndResizeItem burnwnd_rlist[] =
|
||
|
{
|
||
|
{IDC_LIST2, 0x0011},
|
||
|
{IDC_CDINFO, 0x0010},
|
||
|
{IDC_ADD, 0x0101},
|
||
|
{IDC_BURN, 0x0101},
|
||
|
{IDC_CLEAR, 0x0101},
|
||
|
{IDC_BURN_OPTS, 0x0101},
|
||
|
{IDC_CANCEL_BURN, 0x0101},
|
||
|
{IDC_LOGO, 0x1010},
|
||
|
{IDC_BTN_SHOWINFO, 0x1111},
|
||
|
};
|
||
|
static _inline void code(long* v, long* k)
|
||
|
{
|
||
|
unsigned long y = v[0], z = v[1], sum = 0, /* set up */
|
||
|
delta = 0x9e3779b9, n = 32 ; /* key schedule constant*/
|
||
|
|
||
|
while (n-- > 0)
|
||
|
{ /* basic cycle start */
|
||
|
sum += delta;
|
||
|
y += ((z << 4) + k[0]) ^(z + sum) ^((z >> 5) + k[1]);
|
||
|
z += ((y << 4) + k[2]) ^(y + sum) ^((y >> 5) + k[3]); /* end cycle */
|
||
|
}
|
||
|
v[0] = y; v[1] = z;
|
||
|
|
||
|
}
|
||
|
|
||
|
static void startBurn(HWND hwndDlg, char driveletter)
|
||
|
{
|
||
|
g_config->WriteInt(L"cdburnmaxspeed", m_max_speed);
|
||
|
|
||
|
//write the temp playlist to disk
|
||
|
FILE *fp;
|
||
|
char filename[MAX_PATH] = {0}, tp[MAX_PATH] = {0};
|
||
|
|
||
|
pidBurner = 0;
|
||
|
|
||
|
if (!GetTempPathA(sizeof(tp), tp)) lstrcpynA(tp, ".", MAX_PATH);
|
||
|
if (GetTempFileNameA(tp, "BRN", 0, filename))
|
||
|
{
|
||
|
unlink(filename);
|
||
|
StringCchCatA(filename, MAX_PATH, ".m3u8");
|
||
|
}
|
||
|
else lstrcpynA(filename, "brn_tmp.m3u8", MAX_PATH);
|
||
|
|
||
|
fp = fopen(filename, "wt");
|
||
|
if (!fp)
|
||
|
{
|
||
|
wchar_t title[16] = {0};
|
||
|
MessageBox(hwndDlg, WASABI_API_LNGSTRINGW(IDS_ERROR_WRITING_TEMP_BURN_LIST),
|
||
|
WASABI_API_LNGSTRINGW_BUF(IDS_ERROR,title,16), MB_OK);
|
||
|
return ;
|
||
|
}
|
||
|
int idx = LETTERTOINDEX(driveletter);
|
||
|
fprintf(fp, "#EXTM3U\n");
|
||
|
for (int i = 0;i < itemCache[idx].Size;i++)
|
||
|
{
|
||
|
fprintf(fp, "#EXTINF:%d,%s\n", itemCache[idx].Items[i].length, (char *)AutoChar(itemCache[idx].Items[i].title, CP_UTF8));
|
||
|
fprintf(fp, "%s\n", (char *)AutoChar(itemCache[idx].Items[i].filename, CP_UTF8));
|
||
|
}
|
||
|
fclose(fp);
|
||
|
|
||
|
burnCDStruct bcds =
|
||
|
{
|
||
|
m_cdrom,
|
||
|
filename,
|
||
|
hwndDlg,
|
||
|
"",
|
||
|
};
|
||
|
pidBurner = (DWORD)(INT_PTR)SendMessage(plugin.hwndWinampParent, WM_WA_IPC, (WPARAM) & bcds, IPC_BURN_CD);
|
||
|
if (!pidBurner)
|
||
|
{
|
||
|
wchar_t title[16] = {0};
|
||
|
unlink(filename);
|
||
|
MessageBox(hwndDlg, AutoWide(bcds.error), WASABI_API_LNGSTRINGW_BUF(IDS_ERROR,title,16), MB_OK);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static void link_handledraw(HWND hwndDlg, UINT uMsg, WPARAM wParam, LPARAM lParam)
|
||
|
{
|
||
|
if (uMsg == WM_DRAWITEM)
|
||
|
{
|
||
|
DRAWITEMSTRUCT *di = (DRAWITEMSTRUCT *)lParam;
|
||
|
if (di->CtlType == ODT_BUTTON)
|
||
|
{
|
||
|
wchar_t wt[123] = {0};
|
||
|
int y;
|
||
|
RECT r;
|
||
|
HPEN hPen, hOldPen;
|
||
|
GetDlgItemText(hwndDlg, (int)wParam, wt, 123);
|
||
|
|
||
|
// draw text
|
||
|
SetTextColor(di->hDC, (di->itemState & ODS_SELECTED) ? RGB(220, 0, 0) : RGB(0, 0, 220));
|
||
|
r = di->rcItem;
|
||
|
r.left += 2;
|
||
|
DrawText(di->hDC, wt, -1, &r, DT_VCENTER | DT_SINGLELINE);
|
||
|
|
||
|
memset(&r, 0, sizeof(r));
|
||
|
DrawText(di->hDC, wt, -1, &r, DT_SINGLELINE | DT_CALCRECT);
|
||
|
|
||
|
// draw underline
|
||
|
y = di->rcItem.bottom - ((di->rcItem.bottom - di->rcItem.top) - (r.bottom - r.top)) / 2 - 1;
|
||
|
hPen = CreatePen(PS_SOLID, 0, (di->itemState & ODS_SELECTED) ? RGB(220, 0, 0) : RGB(0, 0, 220));
|
||
|
hOldPen = (HPEN) SelectObject(di->hDC, hPen);
|
||
|
MoveToEx(di->hDC, di->rcItem.left + 2, y, NULL);
|
||
|
LineTo(di->hDC, di->rcItem.right + 2 - ((di->rcItem.right - di->rcItem.left) - (r.right - r.left)), y);
|
||
|
SelectObject(di->hDC, hOldPen);
|
||
|
DeleteObject(hPen);
|
||
|
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static void refreshList()
|
||
|
{
|
||
|
if (!m_hwndstatus) return ;
|
||
|
|
||
|
ListView_SetItemCount(m_statuslist.getwnd(), 0);
|
||
|
int idx = LETTERTOINDEX(m_cdrom);
|
||
|
ListView_SetItemCount(m_statuslist.getwnd(), itemCache[idx].Size);
|
||
|
if (itemCache[idx].Size > 0) ListView_RedrawItems(m_statuslist.getwnd(), 0, itemCache[idx].Size - 1);
|
||
|
}
|
||
|
static int m_last_trackpos;
|
||
|
|
||
|
|
||
|
typedef struct _MEDIAINFO
|
||
|
{
|
||
|
CHAR cLetter;
|
||
|
BOOL bInserted;
|
||
|
BOOL bRecordable;
|
||
|
BOOL bRewritable;
|
||
|
BOOL bBlank;
|
||
|
ULONG nSectorsFree;
|
||
|
ULONG nSectorsUsed;
|
||
|
} MEDIAINFO;
|
||
|
|
||
|
static HRESULT GetMediaInfoFromSonic(MEDIAINFO *pmi)
|
||
|
{
|
||
|
HRESULT hr;
|
||
|
|
||
|
hr = S_OK;
|
||
|
|
||
|
char name[]= "cda://X.cda";
|
||
|
char buf2[64] = "";
|
||
|
char buf3[64] = "";
|
||
|
|
||
|
name[6] = pmi->cLetter;
|
||
|
|
||
|
pmi->bInserted = FALSE;
|
||
|
pmi->bRewritable = FALSE;
|
||
|
pmi->nSectorsFree = 0;
|
||
|
pmi->nSectorsUsed = 0;
|
||
|
|
||
|
pmi->bRecordable = TRUE;
|
||
|
getFileInfo(name, "cdtype", buf3, sizeof(buf3));
|
||
|
if (buf3[0] && 0 == lstrcmpA(buf3, "CDRW")) pmi->bRewritable = TRUE;
|
||
|
|
||
|
getFileInfo(name, "cdlengths", buf2, sizeof(buf2));
|
||
|
if (buf2[0])
|
||
|
{
|
||
|
pmi->bInserted = TRUE;
|
||
|
pmi->nSectorsFree = atoi(buf2);
|
||
|
}
|
||
|
|
||
|
return hr;
|
||
|
}
|
||
|
|
||
|
static void CALLBACK FreeAsyncParam(DM_NOTIFY_PARAM *phdr)
|
||
|
{
|
||
|
switch(phdr->opCode)
|
||
|
{
|
||
|
case DMOP_IMAPIINFO:
|
||
|
break;
|
||
|
}
|
||
|
free(phdr);
|
||
|
}
|
||
|
static void FinishSetStatus(HWND hwndDlg, MEDIAINFO *pmi)
|
||
|
{
|
||
|
int freesecs;
|
||
|
if(pmi->bInserted)
|
||
|
{
|
||
|
freesecs = (pmi->nSectorsFree * 2048) / (150 * 1024); //150kb/s as its considered DATA CD at this point in veritas
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
freesecs = 74 * 60; //Default to 74mns CDR
|
||
|
}
|
||
|
|
||
|
m_availsecs = freesecs;
|
||
|
|
||
|
int idx = LETTERTOINDEX(m_cdrom);
|
||
|
int usedlen = 0;
|
||
|
int truncpos = 0;
|
||
|
for (int i = 0;i < itemCache[idx].Size;i++)
|
||
|
{
|
||
|
usedlen += itemCache[idx].Items[i].length;
|
||
|
if (usedlen > m_availsecs)
|
||
|
truncpos++;
|
||
|
}
|
||
|
m_availsecs -= usedlen;
|
||
|
|
||
|
wchar_t status[256] = {0};
|
||
|
if (!pmi->bInserted)
|
||
|
WASABI_API_LNGSTRINGW_BUF(IDS_NO_BLANK_CDR_IN_DRIVE,status,512);
|
||
|
else
|
||
|
{
|
||
|
StringCchPrintf(status, 512, WASABI_API_LNGSTRINGW(IDS_X_CAPACITY_DETAILS),
|
||
|
(pmi->bRewritable) ? L"CD-RW" : L"CD-R" , freesecs / 60, freesecs % 60);
|
||
|
}
|
||
|
|
||
|
wchar_t temp[16] = {0};
|
||
|
StringCchPrintf(status + wcslen(status), 256,
|
||
|
WASABI_API_LNGSTRINGW(IDS_USED_X_X_TRACKS),
|
||
|
usedlen / 60,
|
||
|
usedlen % 60,
|
||
|
itemCache[idx].Size,
|
||
|
WASABI_API_LNGSTRINGW_BUF(itemCache[idx].Size == 1 ? IDS_TRACK : IDS_TRACKS,temp,16));
|
||
|
|
||
|
if (freesecs && pmi->bInserted)
|
||
|
{
|
||
|
if (m_availsecs >= 0) StringCchPrintf(status + wcslen(status), 256, WASABI_API_LNGSTRINGW(IDS_AVAILABLE_X_X),
|
||
|
m_availsecs / 60, m_availsecs % 60);
|
||
|
else StringCchPrintf(status + wcslen(status), 256, WASABI_API_LNGSTRINGW(IDS_X_OVER_CAPACITY_REMOVE_X_TRACKS),
|
||
|
-m_availsecs / 60, -m_availsecs % 60, truncpos);
|
||
|
}
|
||
|
SetDlgItemText(hwndDlg, IDC_CDINFO, status);
|
||
|
m_last_trackpos = -1;
|
||
|
|
||
|
m_is_cdrw = pmi->bRewritable;
|
||
|
ListView_RedrawItems(m_statuslist.getwnd(), 0, itemCache[idx].Size - 1);
|
||
|
}
|
||
|
|
||
|
static void SetStatus(HWND hwndDlg, CHAR cLetter)
|
||
|
{
|
||
|
if (DM_MODE_BURNING == DriveManager_GetDriveMode(cLetter) &&
|
||
|
NULL != (m_burning_other_wnd = cdburn_FindBurningHWND(cLetter)))
|
||
|
{
|
||
|
prevWnd = (HWND)SendMessage(m_burning_other_wnd, WM_BURNUPDATEOWNER, 0, (LPARAM)hwndDlg);
|
||
|
if (prevWnd == hwndDlg) prevWnd = NULL;
|
||
|
DWORD state = (DWORD)(INT_PTR)SendMessage(m_burning_other_wnd, WM_BURNGETSTATUS, BURNSTATUS_STATE, 0);
|
||
|
if (state)
|
||
|
{
|
||
|
SetDlgItemText(hwndDlg, IDC_CDINFO, WASABI_API_LNGSTRINGW(IDS_BURNING));
|
||
|
ShowWindow(GetDlgItem(hwndDlg, IDC_CLEAR), SW_HIDE);
|
||
|
ShowWindow(GetDlgItem(hwndDlg, IDC_ADD), SW_HIDE);
|
||
|
ShowWindow(GetDlgItem(hwndDlg, IDC_BURN), SW_HIDE);
|
||
|
ShowWindow(GetDlgItem(hwndDlg, IDC_BURN_OPTS), SW_SHOWNA);
|
||
|
ShowWindow(GetDlgItem(hwndDlg, IDC_CANCEL_BURN), SW_SHOWNA);
|
||
|
SetDlgItemText(hwndDlg, IDC_CANCEL_BURN, WASABI_API_LNGSTRINGW(IDS_CANCEL_BURN));
|
||
|
m_availsecs = 0;
|
||
|
m_last_trackpos = -1;
|
||
|
m_is_cdrw = 0;
|
||
|
percentCompleted = 0;
|
||
|
UpdateWindow(hwndDlg);
|
||
|
}
|
||
|
|
||
|
SendMessage(hwndDlg, WM_BURNNOTIFY, BURN_STATECHANGED, state);
|
||
|
ShowWindow(m_burning_other_wnd, g_config->ReadInt(L"cdburnstatuswnd", 1) ? SW_SHOWNA : SW_HIDE);
|
||
|
PostMessage(m_burning_other_wnd, WM_BURNCONFIGCHANGED, BURNCFG_HIDEVIEW, !g_config->ReadInt(L"cdburnstatuswnd", 1));
|
||
|
PostMessage(m_burning_other_wnd, WM_BURNCONFIGCHANGED, BURNCFG_AUTOEJECT, g_config->ReadInt(L"cdburnautoeject", 1));
|
||
|
PostMessage(m_burning_other_wnd, WM_BURNCONFIGCHANGED, BURNCFG_ADDTODB, g_config->ReadInt(L"cdburnautoadddb", 0));
|
||
|
PostMessage(m_burning_other_wnd, WM_BURNCONFIGCHANGED, BURNCFG_AUTOCLOSE, g_config->ReadInt(L"cdburnautoclose", 0));
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
BOOL br;
|
||
|
ShowWindow(GetDlgItem(hwndDlg, IDC_CLEAR), SW_SHOWNA);
|
||
|
ShowWindow(GetDlgItem(hwndDlg, IDC_ADD), SW_SHOWNA);
|
||
|
ShowWindow(GetDlgItem(hwndDlg, IDC_BURN), SW_SHOWNA);
|
||
|
ShowWindow(GetDlgItem(hwndDlg, IDC_BURN_OPTS), SW_HIDE);
|
||
|
ShowWindow(GetDlgItem(hwndDlg, IDC_CANCEL_BURN), SW_HIDE);
|
||
|
|
||
|
SetDlgItemText(hwndDlg, IDC_CDINFO, WASABI_API_LNGSTRINGW(IDS_CALCULATING));
|
||
|
|
||
|
UpdateWindow(hwndDlg);
|
||
|
|
||
|
DM_IMAPI_PARAM *pIMAPI = (DM_IMAPI_PARAM*)calloc(1, sizeof(DM_IMAPI_PARAM));
|
||
|
if (pIMAPI)
|
||
|
{
|
||
|
pIMAPI->header.cLetter = cLetter;
|
||
|
pIMAPI->header.callback = (INT_PTR)hwndDlg;
|
||
|
pIMAPI->header.uMsg = WM_EX_OPCOMPLETED;
|
||
|
pIMAPI->header.fnFree = FreeAsyncParam;
|
||
|
pIMAPI->header.fFlags = DMF_QUERYMEDIATYPE | DMF_QUERYMEDIAINFO;
|
||
|
br = DriveManager_GetIMAPIInfo(pIMAPI);
|
||
|
}
|
||
|
else br = FALSE;
|
||
|
if (!br) SetDlgItemText(hwndDlg, IDC_CDINFO, WASABI_API_LNGSTRINGW(IDS_DISC_READ_ERROR));
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static void deleteSelectedItems(HWND hwndDlg, CHAR cLetter)
|
||
|
{
|
||
|
int idx = LETTERTOINDEX(cLetter);
|
||
|
for (int i = itemCache[idx].Size - 1;i >= 0;i--)
|
||
|
{
|
||
|
if (m_statuslist.GetSelected(i))
|
||
|
{
|
||
|
freeRecord(&itemCache[idx].Items[i]);
|
||
|
int l = itemCache[idx].Size - i - 1;
|
||
|
if (l > 0) memcpy(&itemCache[idx].Items[i], &itemCache[idx].Items[i + 1], sizeof(itemRecordW)*l);
|
||
|
itemCache[idx].Size--;
|
||
|
}
|
||
|
}
|
||
|
SetStatus(hwndDlg, cLetter);
|
||
|
refreshList();
|
||
|
}
|
||
|
|
||
|
static void selectAll()
|
||
|
{
|
||
|
int l = m_statuslist.GetCount();
|
||
|
for (int i = 0;i < l;i++) m_statuslist.SetSelected(i);
|
||
|
}
|
||
|
|
||
|
static void playSelectedItems(HWND hwndDlg, int enqueue)
|
||
|
{
|
||
|
int idx = LETTERTOINDEX(m_cdrom);
|
||
|
if (!enqueue) SendMessage(plugin.hwndWinampParent, WM_WA_IPC, 0, IPC_DELETE);
|
||
|
|
||
|
for (int i = 0;i < itemCache[idx].Size;i++)
|
||
|
{
|
||
|
if (!m_statuslist.GetSelected(i)) continue;
|
||
|
|
||
|
//send the file to winamp
|
||
|
COPYDATASTRUCT cds;
|
||
|
cds.dwData = IPC_PLAYFILEW;
|
||
|
cds.lpData = (void *)itemCache[idx].Items[i].filename;
|
||
|
cds.cbData = (DWORD)(sizeof(wchar_t *) * (wcslen(itemCache[idx].Items[i].filename) + 1)); // include space for null char
|
||
|
SendMessageW(plugin.hwndWinampParent, WM_COPYDATA, (WPARAM)NULL, (LPARAM)&cds);
|
||
|
}
|
||
|
|
||
|
if (!enqueue) SendMessageW(plugin.hwndWinampParent, WM_WA_IPC, 0, IPC_STARTPLAY);
|
||
|
}
|
||
|
BOOL CALLBACK CantBurnProc(HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam)
|
||
|
{
|
||
|
switch (msg)
|
||
|
{
|
||
|
case WM_INITDIALOG:
|
||
|
{
|
||
|
wchar_t *message = (wchar_t *)lParam;
|
||
|
|
||
|
// due to quirks with the more common resource editors, is easier to just store the string
|
||
|
// internally only with \n and post-process to be \r\n (as here) so it will appear correctly
|
||
|
// on new lines as is wanted (silly multiline edit controls)
|
||
|
wchar_t tmp2[1024] = {0}, *t2 = tmp2;
|
||
|
while(message && *message && (t2 - tmp2 < 1024))
|
||
|
{
|
||
|
if(*message == L'\n')
|
||
|
{
|
||
|
*t2 = L'\r';
|
||
|
t2 = CharNextW(t2);
|
||
|
}
|
||
|
*t2 = *message;
|
||
|
message = CharNextW(message);
|
||
|
t2 = CharNextW(t2);
|
||
|
}
|
||
|
|
||
|
SetDlgItemText(hwnd, IDC_MESSAGE2, tmp2);
|
||
|
}
|
||
|
return 0;
|
||
|
case WM_COMMAND:
|
||
|
switch (LOWORD(wParam))
|
||
|
{
|
||
|
case IDOK:
|
||
|
EndDialog(hwnd, 0);
|
||
|
break;
|
||
|
case IDCANCEL:
|
||
|
EndDialog(hwnd, -1);
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
HRESULT ResolveShortCut(HWND hwnd, LPCSTR pszShortcutFile, LPSTR pszPath)
|
||
|
{
|
||
|
HRESULT hres;
|
||
|
IShellLinkA* psl;
|
||
|
WIN32_FIND_DATAA wfd;
|
||
|
|
||
|
*pszPath = 0; // assume failure
|
||
|
|
||
|
hres = CoCreateInstance(CLSID_ShellLink, NULL, CLSCTX_INPROC_SERVER,
|
||
|
IID_IShellLinkA, (void **) & psl);
|
||
|
if (SUCCEEDED(hres))
|
||
|
{
|
||
|
IPersistFile* ppf;
|
||
|
|
||
|
hres = psl->QueryInterface(IID_IPersistFile, (void **) & ppf); // OLE 2! Yay! --YO
|
||
|
if (SUCCEEDED(hres))
|
||
|
{
|
||
|
wchar_t wsz[MAX_PATH] = {0};
|
||
|
MultiByteToWideCharSZ(CP_ACP, 0, pszShortcutFile, -1, wsz, MAX_PATH);
|
||
|
|
||
|
hres = ppf->Load(wsz, STGM_READ);
|
||
|
if (SUCCEEDED(hres))
|
||
|
{
|
||
|
hres = psl->Resolve(hwnd, SLR_ANY_MATCH);
|
||
|
if (SUCCEEDED(hres))
|
||
|
{
|
||
|
char szGotPath[MAX_PATH] = {0};
|
||
|
lstrcpynA(szGotPath, pszShortcutFile, MAX_PATH);
|
||
|
hres = psl->GetPath(szGotPath, MAX_PATH, (WIN32_FIND_DATAA *) & wfd,
|
||
|
SLGP_SHORTPATH);
|
||
|
lstrcpynA(pszPath, szGotPath, MAX_PATH);
|
||
|
}
|
||
|
}
|
||
|
ppf->Release();
|
||
|
}
|
||
|
psl->Release();
|
||
|
}
|
||
|
return SUCCEEDED(hres);
|
||
|
}
|
||
|
|
||
|
HRESULT ResolveShortCut(HWND hwnd, LPCWSTR pszShortcutFile, LPWSTR pszPath)
|
||
|
{
|
||
|
HRESULT hres;
|
||
|
IShellLinkW* psl;
|
||
|
WIN32_FIND_DATAW wfd;
|
||
|
|
||
|
*pszPath = 0; // assume failure
|
||
|
|
||
|
hres = CoCreateInstance(CLSID_ShellLink, NULL, CLSCTX_INPROC_SERVER,
|
||
|
IID_IShellLinkW, (void **) & psl);
|
||
|
if (SUCCEEDED(hres))
|
||
|
{
|
||
|
IPersistFile* ppf;
|
||
|
|
||
|
hres = psl->QueryInterface(IID_IPersistFile, (void **) & ppf); // OLE 2! Yay! --YO
|
||
|
if (SUCCEEDED(hres))
|
||
|
{
|
||
|
/*wchar_t wsz[MAX_PATH] = {0};
|
||
|
MultiByteToWideCharSZ(CP_ACP, 0, pszShortcutFile, -1, wsz, MAX_PATH);*/
|
||
|
|
||
|
hres = ppf->Load(pszShortcutFile/*wsz*/, STGM_READ);
|
||
|
if (SUCCEEDED(hres))
|
||
|
{
|
||
|
hres = psl->Resolve(hwnd, SLR_ANY_MATCH);
|
||
|
if (SUCCEEDED(hres))
|
||
|
{
|
||
|
wchar_t szGotPath[MAX_PATH] = {0};
|
||
|
wcsncpy(szGotPath, pszShortcutFile, MAX_PATH);
|
||
|
hres = psl->GetPath(szGotPath, MAX_PATH, (WIN32_FIND_DATAW *) & wfd,
|
||
|
SLGP_SHORTPATH);
|
||
|
wcsncpy(pszPath, szGotPath, MAX_PATH);
|
||
|
}
|
||
|
}
|
||
|
ppf->Release();
|
||
|
}
|
||
|
psl->Release();
|
||
|
}
|
||
|
return SUCCEEDED(hres);
|
||
|
}
|
||
|
|
||
|
static int checkFile(const char *file)
|
||
|
{
|
||
|
//check if the file is supported by winamp
|
||
|
const char *ext = extension(file);
|
||
|
if (!ext || !ext[0]) return 0;
|
||
|
if (strstr(file, "://") && !strstr(file, "cda://")) return 0;
|
||
|
|
||
|
#if 0 // benski> this would be neat to have, but will fail with unicode filenames (which in_mp3 can open anyway)... TODO: make it workable later
|
||
|
HANDLE hFile = CreateFile(file, GENERIC_READ, FILE_SHARE_READ, NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL);
|
||
|
if (INVALID_HANDLE_VALUE == hFile && GetLastError() != ERROR_FILE_NOT_FOUND)
|
||
|
{
|
||
|
wchar_t message[1024] = {0};
|
||
|
StringCchPrintfW(message, 1024, WASABI_API_LNGSTRINGW(IDS_FILE_X_CANNOT_BE_BURNED_REASON_NOT_FOUND), AutoWide(file), AutoWide(ext));
|
||
|
return WASABI_API_DIALOGBOXPARAM(IDD_NOBURN, g_hwnd, CantBurnProc, (LPARAM)message);
|
||
|
}
|
||
|
CloseHandle(hFile);
|
||
|
#endif
|
||
|
|
||
|
char *m_extlist = (char*)SendMessage(plugin.hwndWinampParent, WM_WA_IPC, 0, IPC_GET_EXTLIST);
|
||
|
{
|
||
|
int found = 0;
|
||
|
char *a = m_extlist;
|
||
|
while (a && *a)
|
||
|
{
|
||
|
if (!lstrcmpiA(a, ext))
|
||
|
{
|
||
|
found = 1;
|
||
|
break;
|
||
|
}
|
||
|
a += lstrlenA(a) + 1;
|
||
|
}
|
||
|
GlobalFree((HGLOBAL)m_extlist);
|
||
|
if (!found)
|
||
|
{
|
||
|
wchar_t message[1024] = {0};
|
||
|
StringCchPrintfW(message, 1024, WASABI_API_LNGSTRINGW(IDS_FILE_X_CANNOT_BE_BURNED_REASON_FILETYPE_NOT_REGISTERED), AutoWide(file), AutoWide(ext));
|
||
|
return (int)(INT_PTR)WASABI_API_DIALOGBOXPARAM(IDD_NOBURN, plugin.hwndLibraryParent, CantBurnProc, (LPARAM)message);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
//check for type
|
||
|
char tmp[64] = {0, };
|
||
|
getFileInfo(file, "type", tmp, sizeof(tmp) - 1);
|
||
|
if (tmp[0] && tmp[0] != '0')
|
||
|
{
|
||
|
wchar_t message[1024], temp[128] = {0};
|
||
|
StringCchPrintfW(message, 1024, WASABI_API_LNGSTRINGW(IDS_FILE_X_CANNOT_BE_BURNED_REASON_X),AutoWide(file),
|
||
|
WASABI_API_LNGSTRINGW_BUF((tmp[0] == '1' ? IDS_VIDEO_FILES_CANNOT_BE_BURNED : IDS_NOT_AN_AUDIO_FILE),temp,128));
|
||
|
return (int)(INT_PTR)WASABI_API_DIALOGBOXPARAM(IDD_NOBURN, plugin.hwndLibraryParent, CantBurnProc, (LPARAM)message);
|
||
|
|
||
|
}
|
||
|
|
||
|
// note: this check is NOT meant as any sort of protection.. It simply saves the user the hassle of an error later
|
||
|
if (getFileInfo(file, "burnable", tmp, 64) // most plugins don't support this extended file info, so failure is OK
|
||
|
&& tmp[0] == '0') // if it does support it, then we can check whether or not it's burnable
|
||
|
{
|
||
|
wchar_t message[1024] = {0};
|
||
|
StringCchPrintfW(message, 1024, WASABI_API_LNGSTRINGW(IDS_FILE_CANNOT_BE_BURNED), AutoWide(file));
|
||
|
if (getFileInfo(file, "noburnreason", tmp, 64))
|
||
|
{
|
||
|
StringCchPrintfW(message, 1024,
|
||
|
WASABI_API_LNGSTRINGW(IDS_FILE_X_CANNOT_BE_BURNED_REASON_X),
|
||
|
AutoWide(file), AutoWide(tmp));
|
||
|
}
|
||
|
return (int)(INT_PTR)WASABI_API_DIALOGBOXPARAM(IDD_NOBURN, plugin.hwndLibraryParent, CantBurnProc, (LPARAM)message);
|
||
|
}
|
||
|
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
|
||
|
static int checkFile(const wchar_t *file)
|
||
|
{
|
||
|
//check if the file is supported by winamp
|
||
|
const wchar_t *ext = PathFindExtension(file);
|
||
|
if (!ext || !ext[0]) return 0;
|
||
|
ext++;
|
||
|
if (wcsstr(file, L"://") && !wcsstr(file, L"cda://")) return 0;
|
||
|
|
||
|
#if 0 // benski> this would be neat to have, but will fail with unicode filenames (which in_mp3 can open anyway)... TODO: make it workable later
|
||
|
HANDLE hFile = CreateFile(file, GENERIC_READ, FILE_SHARE_READ, NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL);
|
||
|
if (INVALID_HANDLE_VALUE == hFile && GetLastError() == ERROR_FILE_NOT_FOUND)
|
||
|
{
|
||
|
wchar_t message[1024] = {0};
|
||
|
StringCchPrintfW(message, 1024, WASABI_API_LNGSTRINGW(IDS_FILE_X_CANNOT_BE_BURNED_REASON_NOT_FOUND), file, ext);
|
||
|
return WASABI_API_DIALOGBOXPARAM(IDD_NOBURN, g_hwnd, CantBurnProc, (LPARAM)message);
|
||
|
}
|
||
|
CloseHandle(hFile);
|
||
|
#endif
|
||
|
wchar_t *m_extlist = (wchar_t*)SendMessage(plugin.hwndWinampParent, WM_WA_IPC, 0, IPC_GET_EXTLISTW);
|
||
|
{
|
||
|
int found = 0;
|
||
|
wchar_t *a = m_extlist;
|
||
|
while (a && *a)
|
||
|
{
|
||
|
if (!lstrcmpiW(a, ext))
|
||
|
{
|
||
|
found = 1;
|
||
|
break;
|
||
|
}
|
||
|
a += lstrlenW(a) + 1;
|
||
|
}
|
||
|
GlobalFree((HGLOBAL)m_extlist);
|
||
|
if (!found)
|
||
|
{
|
||
|
wchar_t message[1024] = {0};
|
||
|
StringCchPrintfW(message, 1024, WASABI_API_LNGSTRINGW(IDS_FILE_X_CANNOT_BE_BURNED_REASON_FILETYPE_NOT_REGISTERED), file, ext);
|
||
|
return (int)(INT_PTR)WASABI_API_DIALOGBOXPARAM(IDD_NOBURN, plugin.hwndLibraryParent, CantBurnProc, (LPARAM)message);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
//check for type
|
||
|
wchar_t tmp[64] = {0, };
|
||
|
getFileInfoW(file, L"type", tmp, 64);
|
||
|
if (tmp[0] && tmp[0] != '0')
|
||
|
{
|
||
|
wchar_t message[1024] = {0}, temp[128] = {0};
|
||
|
StringCchPrintfW(message, 1024, WASABI_API_LNGSTRINGW(IDS_FILE_X_CANNOT_BE_BURNED_REASON_X), file,
|
||
|
WASABI_API_LNGSTRINGW_BUF((tmp[0] == '1' ? IDS_VIDEO_FILES_CANNOT_BE_BURNED : IDS_NOT_AN_AUDIO_FILE),temp,128));
|
||
|
return (int)(INT_PTR)WASABI_API_DIALOGBOXPARAM(IDD_NOBURN, plugin.hwndLibraryParent, CantBurnProc, (LPARAM)message);
|
||
|
|
||
|
}
|
||
|
|
||
|
// note: this check is NOT meant as any sort of protection.. It simply saves the user the hassle of an error later
|
||
|
if (getFileInfoW(file, L"burnable", tmp, 64) // most plugins don't support this extended file info, so failure is OK
|
||
|
&& tmp[0] == '0') // if it does support it, then we can check whether or not it's burnable
|
||
|
{
|
||
|
wchar_t message[1024] = {0};
|
||
|
StringCchPrintfW(message, 1024, WASABI_API_LNGSTRINGW(IDS_FILE_CANNOT_BE_BURNED), file);
|
||
|
if (getFileInfoW(file, L"noburnreason", tmp, 64))
|
||
|
{
|
||
|
StringCchPrintfW(message, 1024,
|
||
|
WASABI_API_LNGSTRINGW(IDS_FILE_X_CANNOT_BE_BURNED_REASON_X),
|
||
|
file, tmp);
|
||
|
}
|
||
|
return (int)(INT_PTR)WASABI_API_DIALOGBOXPARAM(IDD_NOBURN, plugin.hwndLibraryParent, CantBurnProc, (LPARAM)message);
|
||
|
}
|
||
|
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
|
||
|
void cdburn_clearBurner(char driveletter)
|
||
|
{
|
||
|
emptyRecordList(&itemCache[LETTERTOINDEX(driveletter)]);
|
||
|
}
|
||
|
void cdburn_addfile(char* file, std::vector<GayString*> *files, api_playlistmanager* plManager, ifc_playlistloadercallback *plCB);
|
||
|
void cdburn_addfile(wchar_t* file, std::vector<GayStringW*> *files, api_playlistmanager* plManager, ifc_playlistloadercallback *plCB);
|
||
|
void cdburn_addfolder(const char* folder, std::vector<GayString*> *files, api_playlistmanager* plManager, ifc_playlistloadercallback *plCB);
|
||
|
void cdburn_addfolder(const wchar_t* folder, std::vector<GayStringW*> *files, api_playlistmanager* plManager, ifc_playlistloadercallback *plCB);
|
||
|
|
||
|
void cdburn_appendFile(char *file, char cLetter)
|
||
|
{
|
||
|
std::vector<GayString*> files;
|
||
|
waServiceFactory *plmFactory = plugin.service->service_getServiceByGuid(api_playlistmanagerGUID);
|
||
|
api_playlistmanager *plManager = (plmFactory) ? (api_playlistmanager*)plmFactory->getInterface() : NULL;
|
||
|
|
||
|
int idx = LETTERTOINDEX(cLetter);
|
||
|
int validFile = 1;
|
||
|
|
||
|
if (itemCache[idx].Size > 255) return;
|
||
|
itemRecordListW newItems = {0, 0, 0};
|
||
|
PLCallBack plCB;
|
||
|
plCB.fileList = &files;
|
||
|
|
||
|
cdburn_addfile(file, &files, (api_playlistmanager*)plManager, (ifc_playlistloadercallback*)&plCB);
|
||
|
|
||
|
size_t x;
|
||
|
|
||
|
for (x = 0; x < files.size(); x ++) // temp record . replace it !!!
|
||
|
{
|
||
|
char *fn = files.at(x)->Get();
|
||
|
|
||
|
validFile = checkFile(fn);
|
||
|
// can't use switch here cause break won't work
|
||
|
if (validFile == -1) // bad file and user cancelled
|
||
|
break;
|
||
|
if (validFile) // bad file, user skipped
|
||
|
{
|
||
|
allocRecordList(&newItems, newItems.Size + 1);
|
||
|
if (!newItems.Alloc) break;
|
||
|
|
||
|
char title[2048] = {0};
|
||
|
basicFileInfoStruct bfis = {fn, 0, 0, title, sizeof(title) - 1,};
|
||
|
SendMessage(plugin.hwndWinampParent, WM_WA_IPC, (WPARAM)&bfis, IPC_GET_BASIC_FILE_INFO);
|
||
|
if (bfis.length > 0)
|
||
|
{
|
||
|
memset((void *)&(newItems.Items[newItems.Size]), 0, sizeof(itemRecordW));
|
||
|
newItems.Items[newItems.Size].filename = AutoWideDup(fn);
|
||
|
newItems.Items[newItems.Size].title = AutoWideDup(title);
|
||
|
newItems.Items[newItems.Size].length = bfis.length;
|
||
|
newItems.Size++;
|
||
|
}
|
||
|
}
|
||
|
delete(files.at(x)->Get());
|
||
|
}
|
||
|
|
||
|
if (validFile != -1)
|
||
|
copyRecordList(&itemCache[idx], &newItems);
|
||
|
refreshList();
|
||
|
if (m_hwndstatus) SetStatus(m_hwndstatus, cLetter);
|
||
|
if (plManager) plmFactory->releaseInterface(plManager);
|
||
|
}
|
||
|
|
||
|
void cdburn_appendFile(wchar_t *file, char cLetter)
|
||
|
{
|
||
|
std::vector<GayStringW*> files;
|
||
|
waServiceFactory *plmFactory = plugin.service->service_getServiceByGuid(api_playlistmanagerGUID);
|
||
|
api_playlistmanager *plManager = (plmFactory) ? (api_playlistmanager*)plmFactory->getInterface() : NULL;
|
||
|
|
||
|
int idx = LETTERTOINDEX(cLetter);
|
||
|
int validFile = 1;
|
||
|
|
||
|
if (itemCache[idx].Size > 255) return;
|
||
|
itemRecordListW newItems = {0, 0, 0};
|
||
|
PLCallBackW plCB;
|
||
|
plCB.fileList = &files;
|
||
|
|
||
|
cdburn_addfile(file, &files, (api_playlistmanager*)plManager, (ifc_playlistloadercallback*)&plCB);
|
||
|
|
||
|
size_t x;
|
||
|
|
||
|
for (x = 0; x < files.size(); x ++) // temp record . replace it !!!
|
||
|
{
|
||
|
const wchar_t *fn = files.at(x)->Get();
|
||
|
|
||
|
validFile = checkFile(fn);
|
||
|
// can't use switch here cause break won't work
|
||
|
if (validFile == -1) // bad file and user cancelled
|
||
|
break;
|
||
|
if (validFile) // bad file, user skipped
|
||
|
{
|
||
|
allocRecordList(&newItems, newItems.Size + 1);
|
||
|
if (!newItems.Alloc) break;
|
||
|
|
||
|
wchar_t title[2048] = {0};
|
||
|
basicFileInfoStructW bfis = {fn, 0, 0, title, ARRAYSIZE(title) - 1,};
|
||
|
SendMessage(plugin.hwndWinampParent, WM_WA_IPC, (WPARAM)&bfis, IPC_GET_BASIC_FILE_INFOW);
|
||
|
if (bfis.length > 0)
|
||
|
{
|
||
|
memset((void *)&(newItems.Items[newItems.Size]), 0, sizeof(itemRecordW));
|
||
|
newItems.Items[newItems.Size].filename = _wcsdup(fn);
|
||
|
newItems.Items[newItems.Size].title = _wcsdup(title);
|
||
|
newItems.Items[newItems.Size].length = bfis.length;
|
||
|
newItems.Size++;
|
||
|
}
|
||
|
}
|
||
|
delete(files.at(x)->Get());
|
||
|
}
|
||
|
|
||
|
if (validFile != -1)
|
||
|
copyRecordList(&itemCache[idx], &newItems);
|
||
|
refreshList();
|
||
|
if (m_hwndstatus) SetStatus(m_hwndstatus, cLetter);
|
||
|
if (plManager) plmFactory->releaseInterface(plManager);
|
||
|
}
|
||
|
void cdburn_addfile(char* file, std::vector<GayString*> *files, api_playlistmanager* plManager, ifc_playlistloadercallback *plCB)
|
||
|
{
|
||
|
if (!_stricmp(extension(file), "lnk"))
|
||
|
{
|
||
|
char temp2[MAX_PATH] = {0};
|
||
|
if (ResolveShortCut(plugin.hwndLibraryParent, file, temp2)) lstrcpynA(file, temp2, MAX_PATH);
|
||
|
else return;
|
||
|
}
|
||
|
|
||
|
if (!_strnicmp(file, "cda://", 6))
|
||
|
{
|
||
|
if (strlen(file) == 7)
|
||
|
{
|
||
|
int n = 0;
|
||
|
char buf2[32] = {0};
|
||
|
getFileInfo(file, "ntracks", buf2, sizeof(buf2));
|
||
|
n = atoi(buf2);
|
||
|
if (n > 0 && n < 256)
|
||
|
{
|
||
|
for (int x = 0; x < n; x ++)
|
||
|
{
|
||
|
char s[64] = {0};
|
||
|
StringCchPrintfA(s, 64, "%s,%d.cda", file, x + 1);
|
||
|
files->push_back(new GayString(s));
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
else files->push_back(new GayString(file));
|
||
|
}
|
||
|
|
||
|
else if (strstr(file, "://"))
|
||
|
{
|
||
|
if (plManager && PLAYLISTMANAGER_SUCCESS != plManager->Load(AutoWide(file), plCB))
|
||
|
{
|
||
|
files->push_back(new GayString(file));
|
||
|
}
|
||
|
}
|
||
|
else if (!lstrcmpA(file + 1, ":\\"))
|
||
|
{
|
||
|
cdburn_addfolder(file, files, plManager, plCB);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
WIN32_FIND_DATAA d = {0};
|
||
|
HANDLE h = FindFirstFileA(file, &d);
|
||
|
if (h != INVALID_HANDLE_VALUE)
|
||
|
{
|
||
|
FindClose(h);
|
||
|
if (d.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY)
|
||
|
{
|
||
|
cdburn_addfolder(file, files, plManager, plCB);
|
||
|
}
|
||
|
if (plManager && PLAYLISTMANAGER_SUCCESS != plManager->Load(AutoWide(file), plCB))
|
||
|
{
|
||
|
files->push_back(new GayString(file));
|
||
|
}
|
||
|
}
|
||
|
else files->push_back(new GayString(file));
|
||
|
}
|
||
|
}
|
||
|
void cdburn_addfile(wchar_t* file, std::vector<GayStringW*> *files, api_playlistmanager* plManager, ifc_playlistloadercallback *plCB)
|
||
|
{
|
||
|
if (!_wcsicmp(extensionW(file), L"lnk"))
|
||
|
{
|
||
|
wchar_t temp2[MAX_PATH] = {0};
|
||
|
if (ResolveShortCut(plugin.hwndLibraryParent, file, temp2)) lstrcpyn(file, temp2, MAX_PATH);
|
||
|
else return;
|
||
|
}
|
||
|
|
||
|
if (!_wcsnicmp(file, L"cda://", 6))
|
||
|
{
|
||
|
if (wcslen(file) == 7)
|
||
|
{
|
||
|
int n = 0;
|
||
|
wchar_t buf2[32] = {0};
|
||
|
getFileInfoW(file, L"ntracks", buf2, sizeof(buf2));
|
||
|
n = _wtoi(buf2);
|
||
|
if (n > 0 && n < 256)
|
||
|
{
|
||
|
for (int x = 0; x < n; x ++)
|
||
|
{
|
||
|
wchar_t s[64] = {0};
|
||
|
StringCchPrintfW(s, 64, L"%s,%d.cda", file, x + 1);
|
||
|
files->push_back(new GayStringW(s));
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
else files->push_back(new GayStringW(file));
|
||
|
}
|
||
|
|
||
|
else if (wcsstr(file, L"://"))
|
||
|
{
|
||
|
if (plManager && PLAYLISTMANAGER_SUCCESS != plManager->Load(file, plCB))
|
||
|
{
|
||
|
files->push_back(new GayStringW(file));
|
||
|
}
|
||
|
}
|
||
|
else if (!lstrcmpW(file + 1, L":\\"))
|
||
|
{
|
||
|
cdburn_addfolder(file, files, plManager, plCB);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
WIN32_FIND_DATAW d = {0};
|
||
|
HANDLE h = FindFirstFileW(file, &d);
|
||
|
if (h != INVALID_HANDLE_VALUE)
|
||
|
{
|
||
|
FindClose(h);
|
||
|
if (d.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY)
|
||
|
{
|
||
|
cdburn_addfolder(file, files, plManager, plCB);
|
||
|
}
|
||
|
if (plManager && PLAYLISTMANAGER_SUCCESS != plManager->Load(file, plCB))
|
||
|
{
|
||
|
files->push_back(new GayStringW(file));
|
||
|
}
|
||
|
}
|
||
|
else files->push_back(new GayStringW(file));
|
||
|
}
|
||
|
}
|
||
|
void cdburn_addfolder(const char* folder, std::vector<GayString*> *files, api_playlistmanager* plManager, ifc_playlistloadercallback *plCB)
|
||
|
{
|
||
|
WIN32_FIND_DATAA d = {0};
|
||
|
char path[MAX_PATH] = {0};
|
||
|
PathCombineA(path, folder, "*");
|
||
|
|
||
|
HANDLE h = FindFirstFileA(path, &d);
|
||
|
if (h == INVALID_HANDLE_VALUE) return;
|
||
|
do
|
||
|
{
|
||
|
if (d.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY)
|
||
|
{
|
||
|
if (0 == lstrcmpA(d.cFileName, ".") || 0 == lstrcmpA(d.cFileName, "..")) continue;
|
||
|
GayString pathNew(folder);
|
||
|
pathNew.Append("\\");
|
||
|
pathNew.Append(d.cFileName);
|
||
|
cdburn_addfolder(pathNew.Get(), files, plManager, plCB);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
GayString file(folder);
|
||
|
file.Append("\\");
|
||
|
file.Append(d.cFileName);
|
||
|
cdburn_addfile(file.Get(), files, plManager, plCB);
|
||
|
}
|
||
|
}
|
||
|
while (FindNextFileA(h, &d));
|
||
|
FindClose(h);
|
||
|
}
|
||
|
void cdburn_addfolder(const wchar_t* folder, std::vector<GayStringW*> *files, api_playlistmanager* plManager, ifc_playlistloadercallback *plCB)
|
||
|
{
|
||
|
WIN32_FIND_DATAW d = {0};
|
||
|
wchar_t path[MAX_PATH] = {0};
|
||
|
PathCombineW(path, folder, L"*");
|
||
|
|
||
|
HANDLE h = FindFirstFileW(path, &d);
|
||
|
if (h == INVALID_HANDLE_VALUE) return;
|
||
|
do
|
||
|
{
|
||
|
if (d.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY)
|
||
|
{
|
||
|
if (0 == lstrcmpW(d.cFileName, L".") || 0 == lstrcmpW(d.cFileName, L"..")) continue;
|
||
|
GayStringW pathNew(folder);
|
||
|
pathNew.Append(L"\\");
|
||
|
pathNew.Append(d.cFileName);
|
||
|
cdburn_addfolder(pathNew.Get(), files, plManager, plCB);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
GayStringW file(folder);
|
||
|
file.Append(L"\\");
|
||
|
file.Append(d.cFileName);
|
||
|
cdburn_addfile((wchar_t*)file.Get(), files, plManager, plCB);
|
||
|
}
|
||
|
}
|
||
|
while (FindNextFileW(h, &d));
|
||
|
FindClose(h);
|
||
|
}
|
||
|
void cdburn_appendItemRecord(itemRecordList *obj, char cLetter)
|
||
|
{
|
||
|
int idx = LETTERTOINDEX(cLetter);
|
||
|
int validFile = 1;
|
||
|
itemRecordListW newItems = {0, 0, 0};
|
||
|
BurnAddStatus_Create(obj->Size);
|
||
|
|
||
|
for (int i = 0;i < obj->Size;i++)
|
||
|
{
|
||
|
validFile = checkFile(obj->Items[i].filename);
|
||
|
if (validFile == -1)
|
||
|
break;
|
||
|
if (validFile)
|
||
|
{
|
||
|
if (newItems.Size > 255) break;
|
||
|
|
||
|
allocRecordList(&newItems, newItems.Size + 1);
|
||
|
if (!newItems.Alloc) return ;
|
||
|
|
||
|
memset((void *)&(newItems.Items[newItems.Size]), 0, sizeof(itemRecordW));
|
||
|
newItems.Items[newItems.Size].filename = AutoWideDup(obj->Items[i].filename);
|
||
|
|
||
|
GayString title;
|
||
|
if (obj->Items[i].artist) title.Append(obj->Items[i].artist);
|
||
|
if (title.Get() && title.Get()[0] && obj->Items[i].title && obj->Items[i].title[0])
|
||
|
title.Append(" - ");
|
||
|
if (obj->Items[i].title) title.Append(obj->Items[i].title);
|
||
|
|
||
|
newItems.Items[newItems.Size].title = AutoWideDup(title.Get());
|
||
|
|
||
|
newItems.Items[newItems.Size].length = obj->Items[i].length;
|
||
|
newItems.Size++;
|
||
|
BurnAddStatus_Step(&newItems);
|
||
|
}
|
||
|
}
|
||
|
BurnAddStatus_Done();
|
||
|
if (validFile != -1)
|
||
|
copyRecordList(&itemCache[idx], &newItems);
|
||
|
refreshList();
|
||
|
if (m_hwndstatus) SetStatus(m_hwndstatus, cLetter);
|
||
|
}
|
||
|
|
||
|
void cdburn_appendItemRecord(itemRecordListW *obj, char cLetter)
|
||
|
{
|
||
|
int idx = LETTERTOINDEX(cLetter);
|
||
|
int validFile = 1;
|
||
|
itemRecordListW newItems = {0, 0, 0};
|
||
|
BurnAddStatus_Create(obj->Size);
|
||
|
|
||
|
for (int i = 0;i < obj->Size;i++)
|
||
|
{
|
||
|
validFile = checkFile(obj->Items[i].filename);
|
||
|
if (validFile == -1)
|
||
|
break;
|
||
|
if (validFile)
|
||
|
{
|
||
|
if (newItems.Size > 255) break;
|
||
|
|
||
|
allocRecordList(&newItems, newItems.Size + 1);
|
||
|
if (!newItems.Alloc) return ;
|
||
|
|
||
|
memset((void *)&(newItems.Items[newItems.Size]), 0, sizeof(itemRecordW));
|
||
|
newItems.Items[newItems.Size].filename = _wcsdup(obj->Items[i].filename);
|
||
|
|
||
|
GayStringW title;
|
||
|
if (obj->Items[i].artist) title.Append(obj->Items[i].artist);
|
||
|
if (title.Get() && title.Get()[0] && obj->Items[i].title && obj->Items[i].title[0])
|
||
|
title.Append(L" - ");
|
||
|
if (obj->Items[i].title) title.Append(obj->Items[i].title);
|
||
|
|
||
|
newItems.Items[newItems.Size].title = _wcsdup(title.Get());
|
||
|
|
||
|
newItems.Items[newItems.Size].length = obj->Items[i].length;
|
||
|
newItems.Size++;
|
||
|
BurnAddStatus_Step(&newItems);
|
||
|
}
|
||
|
}
|
||
|
BurnAddStatus_Done();
|
||
|
if (validFile != -1)
|
||
|
copyRecordList(&itemCache[idx], &newItems);
|
||
|
refreshList();
|
||
|
if (m_hwndstatus) SetStatus(m_hwndstatus, cLetter);
|
||
|
}
|
||
|
|
||
|
static void Shell_Free(void *p)
|
||
|
{
|
||
|
IMalloc *m;
|
||
|
SHGetMalloc(&m);
|
||
|
m->Free(p);
|
||
|
}
|
||
|
|
||
|
HWND cdburn_FindBurningHWND(char cLetter)
|
||
|
{
|
||
|
HWND h = 0;
|
||
|
while (NULL != (h = FindWindowExW(NULL, h, L"#32770", NULL)))
|
||
|
{
|
||
|
if (!GetPropW(h, L"WABURNER")) continue;
|
||
|
if (((char)(INT_PTR)GetPropW(h, L"DRIVE")) == cLetter) return h;
|
||
|
}
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
CHAR cdburn_IsMeBurning(void)
|
||
|
{
|
||
|
if (pidBurner)
|
||
|
{
|
||
|
HWND h = NULL;
|
||
|
DWORD pid;
|
||
|
while (NULL != (h = FindWindowExW(NULL, h, L"#32770", NULL)))
|
||
|
{
|
||
|
if (GetPropW(h, L"WABURNER") && GetWindowThreadProcessId(h, &pid) && pid == pidBurner)
|
||
|
return (CHAR)(INT_PTR)GetPropW(h, L"DRIVE");
|
||
|
}
|
||
|
}
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
static void NotifyInfoWindow(HWND hwnd, LPCWSTR pszFileName, BOOL bForceRefresh)
|
||
|
{
|
||
|
HWND hwndParent;
|
||
|
hwndParent = GetParent(hwnd);
|
||
|
if (hwndParent) SendMessageW(hwndParent, WM_SHOWFILEINFO,
|
||
|
(WPARAM)((bForceRefresh) ? WISF_FORCE : WISF_NORMAL),
|
||
|
(LPARAM)pszFileName);
|
||
|
}
|
||
|
|
||
|
static void moveSelItemsUp()
|
||
|
{
|
||
|
if (DM_MODE_BURNING == DriveManager_GetDriveMode(m_cdrom)) return;
|
||
|
|
||
|
int idx = LETTERTOINDEX(m_cdrom);
|
||
|
for (int i = 0;i < itemCache[idx].Size;i++)
|
||
|
{
|
||
|
if (m_statuslist.GetSelected(i))
|
||
|
{
|
||
|
//swap the 2 items
|
||
|
if (i > 0)
|
||
|
{
|
||
|
itemRecordW tmp = itemCache[idx].Items[i];
|
||
|
itemCache[idx].Items[i] = itemCache[idx].Items[i - 1];
|
||
|
itemCache[idx].Items[i - 1] = tmp;
|
||
|
ListView_SetItemState(m_statuslist.getwnd(), i - 1, LVIS_SELECTED, LVIS_SELECTED);
|
||
|
ListView_SetItemState(m_statuslist.getwnd(), i, 0, LVIS_SELECTED);
|
||
|
ListView_RedrawItems(m_statuslist.getwnd(), i - 1, i);
|
||
|
if (ListView_GetItemState(m_statuslist.getwnd(), i, LVIS_FOCUSED))
|
||
|
{
|
||
|
ListView_SetItemState(m_statuslist.getwnd(), i - 1, LVIS_FOCUSED, LVIS_FOCUSED);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static void moveSelItemsDown()
|
||
|
{
|
||
|
if (DM_MODE_BURNING == DriveManager_GetDriveMode(m_cdrom)) return ;
|
||
|
|
||
|
int idx = LETTERTOINDEX(m_cdrom);
|
||
|
for (int i = itemCache[idx].Size - 1;i >= 0;i--)
|
||
|
{
|
||
|
if (m_statuslist.GetSelected(i))
|
||
|
{
|
||
|
//swap the 2 items
|
||
|
if (i < (itemCache[idx].Size - 1))
|
||
|
{
|
||
|
itemRecordW tmp = itemCache[idx].Items[i];
|
||
|
itemCache[idx].Items[i] = itemCache[idx].Items[i + 1];
|
||
|
itemCache[idx].Items[i + 1] = tmp;
|
||
|
ListView_SetItemState(m_statuslist.getwnd(), i + 1, LVIS_SELECTED, LVIS_SELECTED);
|
||
|
ListView_SetItemState(m_statuslist.getwnd(), i, 0, LVIS_SELECTED);
|
||
|
ListView_RedrawItems(m_statuslist.getwnd(), i, i + 1);
|
||
|
if (ListView_GetItemState(m_statuslist.getwnd(), i, LVIS_FOCUSED))
|
||
|
{
|
||
|
ListView_SetItemState(m_statuslist.getwnd(), i + 1, LVIS_FOCUSED, LVIS_FOCUSED);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
int g_burn_hack_startburn;
|
||
|
|
||
|
void OnBurnDlgInit(HWND hwndDlg, LPARAM lParam)
|
||
|
{
|
||
|
m_hwndstatus = hwndDlg;
|
||
|
m_is_cdrw = 0;
|
||
|
m_dragging = 0;
|
||
|
|
||
|
m_cdrom = (char)lParam;
|
||
|
|
||
|
SendMessageW(GetParent(hwndDlg), WM_COMMAND, MAKEWPARAM(IDC_BTN_SHOWINFO, BN_EX_GETTEXT), (LPARAM)GetDlgItem(hwndDlg, IDC_BTN_SHOWINFO));
|
||
|
|
||
|
m_statuslist.setwnd(GetDlgItem(hwndDlg, IDC_LIST2));
|
||
|
m_statuslist.AddCol(WASABI_API_LNGSTRINGW(IDS_TRACK_NUMBER), g_view_metaconf->ReadInt(L"col_track", 60));
|
||
|
m_statuslist.AddCol(WASABI_API_LNGSTRINGW(IDS_TITLE), g_view_metaconf->ReadInt(L"col_title", 200));
|
||
|
m_statuslist.AddCol(WASABI_API_LNGSTRINGW(IDS_LENGTH), g_view_metaconf->ReadInt(L"col_len", 80));
|
||
|
m_statuslist.AddCol(WASABI_API_LNGSTRINGW(IDS_STATUS), g_view_metaconf->ReadInt(L"col_status", 200));
|
||
|
|
||
|
childSizer.Init(hwndDlg, burnwnd_rlist, sizeof(burnwnd_rlist) / sizeof(burnwnd_rlist[0]));
|
||
|
|
||
|
if(m_statuslist.getwnd())
|
||
|
{
|
||
|
MLSKINWINDOW sw;
|
||
|
|
||
|
sw.hwndToSkin = m_statuslist.getwnd();
|
||
|
sw.skinType = SKINNEDWND_TYPE_LISTVIEW;
|
||
|
sw.style = SWLVS_FULLROWSELECT | SWLVS_DOUBLEBUFFER | SWLVS_ALTERNATEITEMS | SWS_USESKINFONT | SWS_USESKINCOLORS | SWS_USESKINCURSORS;
|
||
|
MLSkinWindow(plugin.hwndLibraryParent, &sw);
|
||
|
}
|
||
|
|
||
|
refreshList();
|
||
|
|
||
|
// this will make sure that we've got the cddb logo shown even when using a localised version
|
||
|
HANDLE hPrev = (HANDLE) SendDlgItemMessage(hwndDlg,IDC_LOGO,STM_SETIMAGE,IMAGE_BITMAP,
|
||
|
(LPARAM)LoadImage(plugin.hDllInstance,MAKEINTRESOURCE(IDB_LISTITEM_CDDRIVE),
|
||
|
IMAGE_BITMAP,0,0, 0));
|
||
|
if (hPrev) DeleteObject(hPrev);
|
||
|
|
||
|
NotifyInfoWindow(hwndDlg, NULL, TRUE); // ignore cache
|
||
|
SetStatus(hwndDlg, m_cdrom);
|
||
|
|
||
|
if (g_burn_hack_startburn)
|
||
|
{
|
||
|
g_burn_hack_startburn = 0;
|
||
|
PostMessage(hwndDlg, WM_COMMAND, IDC_BURN, 0);
|
||
|
}
|
||
|
|
||
|
}
|
||
|
|
||
|
void OnBurnNotify(HWND hwndDlg, DWORD notification, DWORD param)
|
||
|
{
|
||
|
switch (notification)
|
||
|
{
|
||
|
case BURN_READY:
|
||
|
SetStatus(hwndDlg, m_cdrom);
|
||
|
break;
|
||
|
case BURN_STATECHANGED:
|
||
|
{
|
||
|
wchar_t title[512] = {0};
|
||
|
const wchar_t *buf = NULL;
|
||
|
switch (param)
|
||
|
{
|
||
|
case BURNERPLAYLIST_BURNCANCELING:
|
||
|
SetDlgItemText(hwndDlg, IDC_CANCEL_BURN, WASABI_API_LNGSTRINGW(IDS_CANCELLING));
|
||
|
buf = WASABI_API_LNGSTRINGW_BUF(IDS_BURNING_AUDIO_CANCELLING,title,512);
|
||
|
break;
|
||
|
case BURNERPLAYLIST_BURNFINISHING:
|
||
|
buf = WASABI_API_LNGSTRINGW_BUF(IDS_BURNING_AUDIO_FINISHING,title,512);
|
||
|
break;
|
||
|
case BURNERPLAYLIST_DECODEFINISHED:
|
||
|
buf = WASABI_API_LNGSTRINGW_BUF(IDS_BURNING_AUDIO_DATA_PREP_FINISHED,title,512);
|
||
|
break;
|
||
|
case BURNERPLAYLIST_LICENSINGSTARTING:
|
||
|
buf = WASABI_API_LNGSTRINGW_BUF(IDS_BURNING_AUDIO_VERIFYING_FILES,title,512);
|
||
|
break;
|
||
|
case BURNERPLAYLIST_LICENSINGFINISHED:
|
||
|
buf = WASABI_API_LNGSTRINGW_BUF(IDS_BURNING_AUDIO_VERIFICATION_COMPLETED,title,512);
|
||
|
break;
|
||
|
case BURNERPLAYLIST_BURNPROGRESS:
|
||
|
wchar_t buf2[256] = {0};
|
||
|
switch (SendMessage(m_burning_other_wnd, WM_BURNGETSTATUS, BURNSTATUS_ERROR, 0))
|
||
|
{
|
||
|
case BURNERPLAYLIST_WRITELEADIN:
|
||
|
buf = WASABI_API_LNGSTRINGW_BUF(IDS_OPENING_DISC_WRITING_LEAD_IN,buf2,256);
|
||
|
break;
|
||
|
case BURNERPLAYLIST_WRITELEADOUT:
|
||
|
buf = WASABI_API_LNGSTRINGW_BUF(IDS_CLOSING_DISC_WRITING_LEAD_OUT,buf2,256);
|
||
|
break;
|
||
|
default: break;
|
||
|
}
|
||
|
if (buf)
|
||
|
{
|
||
|
int percent = (int)(INT_PTR)SendMessage(m_burning_other_wnd, WM_BURNGETSTATUS, BURNSTATUS_PROGRESS, 0);
|
||
|
percentCompleted = max(percent, percentCompleted);
|
||
|
StringCchPrintf(title, 512, WASABI_API_LNGSTRINGW(IDS_BURNING_AUDIO_CURRENT_OPERATION), percentCompleted, buf);
|
||
|
}
|
||
|
break;
|
||
|
}
|
||
|
if (buf) SetDlgItemText(hwndDlg, IDC_CDINFO, title);
|
||
|
}
|
||
|
break;
|
||
|
case BURN_ITEMSTATECHANGED:
|
||
|
ListView_RedrawItems(m_statuslist.getwnd(), param, param);
|
||
|
break;
|
||
|
case BURN_ITEMDECODEPROGRESS:
|
||
|
ListView_RedrawItems(m_statuslist.getwnd(), param, param);
|
||
|
{
|
||
|
wchar_t title[512] = {0};
|
||
|
int percent = (int)(INT_PTR)SendMessage(m_burning_other_wnd, WM_BURNGETSTATUS, BURNSTATUS_PROGRESS, 0);
|
||
|
percentCompleted = max(percent, percentCompleted);
|
||
|
StringCchPrintf(title, 512, WASABI_API_LNGSTRINGW(IDS_BURNING_AUDIO_CD_PREP_DATA), percentCompleted);
|
||
|
SetDlgItemText(hwndDlg, IDC_CDINFO, title);
|
||
|
}
|
||
|
break;
|
||
|
case BURN_ITEMBURNPROGRESS:
|
||
|
ListView_RedrawItems(m_statuslist.getwnd(), param, param);
|
||
|
{
|
||
|
wchar_t title[512] = {0};
|
||
|
int percent = (int)(INT_PTR)SendMessage(m_burning_other_wnd, WM_BURNGETSTATUS, BURNSTATUS_PROGRESS, 0);
|
||
|
percentCompleted = max(percent, percentCompleted);
|
||
|
StringCchPrintf(title, 512, WASABI_API_LNGSTRINGW(IDS_BURNING_AUDIO_BURNING_DATA), percentCompleted);
|
||
|
SetDlgItemText(hwndDlg, IDC_CDINFO, title);
|
||
|
}
|
||
|
break;
|
||
|
case BURN_WORKING:
|
||
|
ListView_RedrawItems(m_statuslist.getwnd(), 0, ListView_GetItemCount(m_statuslist.getwnd()));
|
||
|
break;
|
||
|
case BURN_FINISHED:
|
||
|
{
|
||
|
wchar_t buf1[128] = {0}, closeStr[16] = {0};
|
||
|
GetDlgItemText(hwndDlg, IDC_CANCEL_BURN, buf1, ARRAYSIZE(buf1));
|
||
|
if (lstrcmpi(buf1, WASABI_API_LNGSTRINGW_BUF(IDS_CLOSE,closeStr,16)))
|
||
|
SetDlgItemText(hwndDlg, IDC_CANCEL_BURN, closeStr);
|
||
|
wchar_t buf[128] = {0};
|
||
|
switch (param)
|
||
|
{
|
||
|
case BURNERPLAYLIST_SUCCESS:
|
||
|
WASABI_API_LNGSTRINGW_BUF(IDS_AUDIO_CD_BURNED_SUCCESSFULLY,buf,128);
|
||
|
break;
|
||
|
case BURNERPLAYLIST_ABORTED:
|
||
|
WASABI_API_LNGSTRINGW_BUF(IDS_BURN_ABORTED_BY_USER,buf,128);
|
||
|
break;
|
||
|
default:
|
||
|
WASABI_API_LNGSTRINGW_BUF(IDS_BURNING_FAILED,buf,128);
|
||
|
break;
|
||
|
}
|
||
|
StringCchPrintf(buf1, 128, WASABI_API_LNGSTRINGW(IDS_BURNING_COMPLETED_STATUS_X), buf);
|
||
|
SetDlgItemText(hwndDlg, IDC_CDINFO, buf1);
|
||
|
EnableWindow(GetDlgItem(hwndDlg, IDC_BURN), TRUE);
|
||
|
EnableWindow(GetDlgItem(hwndDlg, IDC_CANCEL_BURN), TRUE);
|
||
|
}
|
||
|
break;
|
||
|
case BURN_DESTROYED:
|
||
|
EnableWindow(GetDlgItem(hwndDlg, IDC_BURN), TRUE);
|
||
|
EnableWindow(GetDlgItem(hwndDlg, IDC_CANCEL_BURN), TRUE);
|
||
|
m_burning_other_wnd = NULL;
|
||
|
SetStatus(hwndDlg, m_cdrom);
|
||
|
break;
|
||
|
case BURN_CONFIGCHANGED:
|
||
|
switch (LOWORD(param))
|
||
|
{
|
||
|
case BURNCFG_AUTOCLOSE:
|
||
|
g_config->WriteInt(L"cdburnautoclose", HIWORD(param));
|
||
|
break;
|
||
|
case BURNCFG_AUTOEJECT:
|
||
|
g_config->WriteInt(L"cdburnautoeject", HIWORD(param));
|
||
|
break;
|
||
|
case BURNCFG_ADDTODB:
|
||
|
g_config->WriteInt(L"cdburnautoadddb", HIWORD(param));
|
||
|
break;
|
||
|
case BURNCFG_HIDEVIEW:
|
||
|
g_config->WriteInt(L"cdburnstatuswnd", !HIWORD(param));
|
||
|
break;
|
||
|
}
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static int CALLBACK WINAPI BrowseCallbackProc(HWND hwnd, UINT uMsg, LPARAM lParam, LPARAM lpData)
|
||
|
{
|
||
|
switch (uMsg)
|
||
|
{
|
||
|
case BFFM_INITIALIZED:
|
||
|
{
|
||
|
SendMessageW(hwnd, BFFM_SETSELECTIONW, 1, (LPARAM)WASABI_API_APP->path_getWorkingPath());
|
||
|
|
||
|
// this is not nice but it fixes the selection not working correctly on all OSes
|
||
|
EnumChildWindows(hwnd, browseEnumProc, 0);
|
||
|
}
|
||
|
}
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
wchar_t* BuildFilterList(void)
|
||
|
{
|
||
|
static wchar_t fileExtensionsString[128] = {L"*.*"}; // "All files\0*.*\0\0"
|
||
|
wchar_t *temp=fileExtensionsString+lstrlenW(fileExtensionsString) +1;
|
||
|
lstrcpynW(temp, WASABI_API_LNGSTRINGW(IDS_ALL_FILES), 128);
|
||
|
*(temp = temp + lstrlenW(temp) + 1) = 0;
|
||
|
return fileExtensionsString;
|
||
|
}
|
||
|
|
||
|
|
||
|
static void CALLBACK Window_TimerProc(HWND hwnd, UINT uMsg, UINT_PTR idEvent, DWORD dwTime)
|
||
|
{
|
||
|
HWND hwndList;
|
||
|
int index, driveIdx;
|
||
|
wchar_t *pszFileName;
|
||
|
|
||
|
switch(idEvent)
|
||
|
{
|
||
|
case TIMER_NOTIFYINFO_ID:
|
||
|
KillTimer(hwnd, TIMER_NOTIFYINFO_ID);
|
||
|
hwndList = GetDlgItem(hwnd, IDC_LIST2);
|
||
|
|
||
|
driveIdx = LETTERTOINDEX(m_cdrom);
|
||
|
index = (hwndList) ? (INT)SendMessage(hwndList, LVM_GETNEXTITEM, (WPARAM)-1, (LPARAM)LVNI_FOCUSED) : -1;
|
||
|
pszFileName = (index >= 0 && index < itemCache[driveIdx].Size) ? itemCache[driveIdx].Items[index].filename : NULL;
|
||
|
NotifyInfoWindow(hwnd, pszFileName, FALSE);
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static void ListView_OnItemChanged(HWND hwndDlg, NMLISTVIEW *pnmv)
|
||
|
{
|
||
|
if (LVIF_STATE & pnmv->uChanged)
|
||
|
{
|
||
|
if ((LVIS_FOCUSED & pnmv->uOldState) != (LVIS_FOCUSED & pnmv->uNewState))
|
||
|
{
|
||
|
KillTimer(hwndDlg, TIMER_NOTIFYINFO_ID);
|
||
|
SetTimer(hwndDlg, TIMER_NOTIFYINFO_ID, TIMER_NOTIFYINFO_DELAY, Window_TimerProc);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
static void Window_OnQueryInfo(HWND hwnd)
|
||
|
{
|
||
|
KillTimer(hwnd, TIMER_NOTIFYINFO_ID);
|
||
|
NotifyInfoWindow(hwnd, NULL, TRUE);
|
||
|
SetTimer(hwnd, TIMER_NOTIFYINFO_ID, TIMER_NOTIFYINFO_DELAY, Window_TimerProc);
|
||
|
}
|
||
|
|
||
|
static void Window_OnOperationCompleted(HWND hwnd, DM_NOTIFY_PARAM *phdr)
|
||
|
{
|
||
|
MEDIAINFO mediaInfo;
|
||
|
|
||
|
if (phdr->cLetter != m_cdrom) return;
|
||
|
|
||
|
ZeroMemory(&mediaInfo, sizeof(MEDIAINFO));
|
||
|
mediaInfo.cLetter = m_cdrom;
|
||
|
|
||
|
switch(phdr->opCode)
|
||
|
{
|
||
|
case DMOP_IMAPIINFO:
|
||
|
if (S_OK == phdr->result)
|
||
|
{
|
||
|
DM_IMAPI_PARAM *pIMAPI = (DM_IMAPI_PARAM*)phdr;
|
||
|
|
||
|
if ((0 != pIMAPI->fMediaType && 0 != pIMAPI->fMediaFlags))
|
||
|
{
|
||
|
mediaInfo.bInserted = TRUE;
|
||
|
if (MEDIA_WRITABLE & pIMAPI->fMediaFlags) mediaInfo.bRecordable = TRUE;
|
||
|
if (MEDIA_RW & pIMAPI->fMediaFlags) mediaInfo.bRewritable = TRUE;
|
||
|
if (MEDIA_BLANK & pIMAPI->fMediaFlags) mediaInfo.bBlank = TRUE;
|
||
|
mediaInfo.nSectorsFree = pIMAPI->ulFreeBlocks;
|
||
|
mediaInfo.nSectorsUsed = pIMAPI->ulNextWritable;
|
||
|
}
|
||
|
}
|
||
|
else GetMediaInfoFromSonic(&mediaInfo);
|
||
|
FinishSetStatus(hwnd, &mediaInfo);
|
||
|
return;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static INT_PTR CALLBACK DlgProc(HWND hwndDlg, UINT uMsg, WPARAM wParam, LPARAM lParam)
|
||
|
{
|
||
|
|
||
|
INT_PTR a = dialogSkinner.Handle(hwndDlg, uMsg, wParam, lParam); if (a) return a;
|
||
|
switch (uMsg)
|
||
|
{
|
||
|
case WM_SIZE:
|
||
|
if (wParam != SIZE_MINIMIZED)
|
||
|
{
|
||
|
childSizer.Resize(hwndDlg, burnwnd_rlist, sizeof(burnwnd_rlist) / sizeof(burnwnd_rlist[0]));
|
||
|
}
|
||
|
break;
|
||
|
case WM_BURNNOTIFY:
|
||
|
OnBurnNotify(hwndDlg, (DWORD)wParam, (DWORD)lParam);
|
||
|
PostMessage(prevWnd, uMsg, wParam, lParam);
|
||
|
break;
|
||
|
case WM_INITDIALOG: OnBurnDlgInit(hwndDlg, lParam); return 0;
|
||
|
|
||
|
case WM_COMMAND:
|
||
|
switch (LOWORD(wParam))
|
||
|
{
|
||
|
// link is dead so disabling for the time being
|
||
|
/*case IDC_LOGO:
|
||
|
if (HIWORD(wParam) == BN_CLICKED)
|
||
|
ShellExecute(hwndDlg, L"open", L"http://estore.sonic.com/redirect.asp?id=spaol110103", NULL, L".", 0);
|
||
|
break;*/
|
||
|
case IDC_BURN_OPTS:
|
||
|
{
|
||
|
RECT r;
|
||
|
HMENU menu = GetSubMenu(g_context_menus, 6);
|
||
|
GetWindowRect((HWND)lParam, &r);
|
||
|
CheckMenuItem(menu, ID_RIPOPTIONS_RIPPINGSTATUSWINDOW, g_config->ReadInt(L"cdburnstatuswnd", 0) ? MF_CHECKED : MF_UNCHECKED);
|
||
|
CheckMenuItem(menu, ID_RIPOPTIONS_EJECTCDWHENCOMPLETED, g_config->ReadInt(L"cdburnautoeject", 1) ? MF_CHECKED : MF_UNCHECKED);
|
||
|
CheckMenuItem(menu, ID_BURNOPTIONS_ADDCDTITLESTOLOCALCDDBCACHE, g_config->ReadInt(L"cdburnautoadddb", 1) ? MF_CHECKED : MF_UNCHECKED);
|
||
|
CheckMenuItem(menu, ID_RIPOPTIONS_CLOSEVIEWWHENCOMPLETE, g_config->ReadInt(L"cdburnautoclose", 0) ? MF_CHECKED : MF_UNCHECKED);
|
||
|
|
||
|
int x = Menu_TrackPopup(plugin.hwndLibraryParent, menu,
|
||
|
TPM_RIGHTBUTTON | TPM_LEFTBUTTON | TPM_BOTTOMALIGN |
|
||
|
TPM_LEFTALIGN | TPM_NONOTIFY | TPM_RETURNCMD,
|
||
|
r.left, r.top, hwndDlg, NULL);
|
||
|
int val = 0, msgid;
|
||
|
switch (x)
|
||
|
{
|
||
|
case ID_RIPOPTIONS_RIPPINGSTATUSWINDOW:
|
||
|
val = g_config->ReadInt(L"cdburnstatuswnd", 0);
|
||
|
g_config->WriteInt(L"cdburnstatuswnd", !val);
|
||
|
msgid = BURNCFG_HIDEVIEW;
|
||
|
break;
|
||
|
case ID_RIPOPTIONS_EJECTCDWHENCOMPLETED:
|
||
|
val = !g_config->ReadInt(L"cdburnautoeject", 1);
|
||
|
g_config->WriteInt(L"cdburnautoeject", val);
|
||
|
msgid = BURNCFG_AUTOEJECT;
|
||
|
break;
|
||
|
case ID_BURNOPTIONS_ADDCDTITLESTOLOCALCDDBCACHE:
|
||
|
val = !g_config->ReadInt(L"cdburnautoadddb", 0);
|
||
|
g_config->WriteInt(L"cdburnautoadddb", val);
|
||
|
msgid = BURNCFG_ADDTODB;
|
||
|
break;
|
||
|
case ID_RIPOPTIONS_CLOSEVIEWWHENCOMPLETE:
|
||
|
val = !g_config->ReadInt(L"cdburnautoclose", 0);
|
||
|
g_config->WriteInt(L"cdburnautoclose", val);
|
||
|
msgid = BURNCFG_AUTOCLOSE;
|
||
|
break;
|
||
|
default: msgid = 0; break;
|
||
|
}
|
||
|
if (msgid)
|
||
|
{
|
||
|
HWND h;
|
||
|
h = cdburn_FindBurningHWND(m_cdrom);
|
||
|
if (h)
|
||
|
{
|
||
|
PostMessage(h, WM_BURNCONFIGCHANGED, msgid, val);
|
||
|
if (BURNCFG_HIDEVIEW == msgid) ShowWindow(h, val ? SW_HIDE : SW_SHOW);
|
||
|
}
|
||
|
}
|
||
|
Sleep(100);
|
||
|
MSG msg;
|
||
|
while (PeekMessage(&msg, NULL, WM_KEYFIRST, WM_KEYLAST, PM_REMOVE)); //eat return
|
||
|
}
|
||
|
return 0;
|
||
|
case IDC_ADD:
|
||
|
if (DM_MODE_BURNING != DriveManager_GetDriveMode(m_cdrom))
|
||
|
{
|
||
|
RECT r;
|
||
|
GetWindowRect((HWND)lParam, &r);
|
||
|
int x = Menu_TrackPopup(plugin.hwndLibraryParent, GetSubMenu(g_context_menus, 3),
|
||
|
TPM_RIGHTBUTTON | TPM_LEFTBUTTON | TPM_BOTTOMALIGN |
|
||
|
TPM_LEFTALIGN | TPM_NONOTIFY | TPM_RETURNCMD,
|
||
|
r.left, r.top, hwndDlg, NULL);
|
||
|
switch (x)
|
||
|
{
|
||
|
case ID_BURNADDMENU_FILES:
|
||
|
{
|
||
|
OPENFILENAMEW l = {sizeof(l), };
|
||
|
wchar_t *temp;
|
||
|
const int len = 256 * 1024 - 128;
|
||
|
wchar_t *m_extlist = 0;
|
||
|
m_extlist = (wchar_t*)SendMessage(plugin.hwndWinampParent, WM_WA_IPC, 1, IPC_GET_EXTLISTW);
|
||
|
if ((int)(INT_PTR)m_extlist == 1) m_extlist = 0;
|
||
|
|
||
|
temp = (wchar_t *)GlobalAlloc(GPTR, len);
|
||
|
l.hwndOwner = hwndDlg;
|
||
|
l.lpstrFilter = m_extlist ? m_extlist : BuildFilterList();
|
||
|
l.lpstrFile = temp;
|
||
|
l.nMaxFile = len - 1;
|
||
|
l.lpstrTitle = WASABI_API_LNGSTRINGW(IDS_ADD_FILES_TO_BURNING_LIST);
|
||
|
l.lpstrDefExt = L"";
|
||
|
l.lpstrInitialDir = WASABI_API_APP->path_getWorkingPath();
|
||
|
|
||
|
l.Flags = OFN_HIDEREADONLY | OFN_EXPLORER | OFN_ALLOWMULTISELECT;
|
||
|
if (GetOpenFileNameW(&l))
|
||
|
{
|
||
|
wchar_t newCurPath[MAX_PATH] = {0};
|
||
|
GetCurrentDirectoryW(MAX_PATH, newCurPath);
|
||
|
WASABI_API_APP->path_setWorkingPath(newCurPath);
|
||
|
|
||
|
if (temp[wcslen(temp) + 1])
|
||
|
{
|
||
|
wchar_t buf[MAX_PATH] = {0};
|
||
|
wchar_t *p = temp;
|
||
|
wchar_t *path = p;
|
||
|
p += wcslen(p) + 1;
|
||
|
while (p && *p)
|
||
|
{
|
||
|
if (*path)
|
||
|
StringCchPrintfW(buf, MAX_PATH, L"%s%s%s", path, path[wcslen(path) - 1] == '\\' ? L"" : L"\\" , p);
|
||
|
else
|
||
|
StringCchPrintfW(buf, MAX_PATH, L"%s", p);
|
||
|
|
||
|
cdburn_appendFile(buf, m_cdrom);
|
||
|
p += wcslen(p) + 1;
|
||
|
}
|
||
|
}
|
||
|
else
|
||
|
cdburn_appendFile(temp, m_cdrom);
|
||
|
}
|
||
|
GlobalFree(temp);
|
||
|
if (m_extlist) GlobalFree((HGLOBAL)m_extlist);
|
||
|
SetStatus(hwndDlg, m_cdrom);
|
||
|
}
|
||
|
break;
|
||
|
case ID_BURNADDMENU_FOLDER:
|
||
|
{
|
||
|
BROWSEINFOW bi = {0};
|
||
|
wchar_t name[MAX_PATH] = {0};
|
||
|
bi.hwndOwner = hwndDlg;
|
||
|
bi.pszDisplayName = name;
|
||
|
bi.lpszTitle = WASABI_API_LNGSTRINGW(IDS_CHOOSE_A_FOLDER_TO_ADD_TO_BURNING_LIST);
|
||
|
bi.ulFlags = BIF_RETURNONLYFSDIRS | BIF_NEWDIALOGSTYLE;
|
||
|
bi.lpfn = BrowseCallbackProc;
|
||
|
ITEMIDLIST *idlist = SHBrowseForFolderW(&bi);
|
||
|
if (idlist)
|
||
|
{
|
||
|
wchar_t path[MAX_PATH] = {0};
|
||
|
SHGetPathFromIDListW(idlist, path);
|
||
|
Shell_Free(idlist);
|
||
|
cdburn_appendFile(path, m_cdrom);
|
||
|
SetStatus(hwndDlg, m_cdrom);
|
||
|
}
|
||
|
}
|
||
|
break;
|
||
|
case ID_BURNADDMENU_CURRENTPLAYLIST:
|
||
|
{
|
||
|
int plsize = (int)(INT_PTR)SendMessage(plugin.hwndWinampParent, WM_WA_IPC, 0, IPC_GETLISTLENGTH);
|
||
|
for (int i = 0;i < plsize;i++)
|
||
|
{
|
||
|
wchar_t *name = (wchar_t *)SendMessage(plugin.hwndWinampParent, WM_WA_IPC, i, IPC_GETPLAYLISTFILEW);
|
||
|
cdburn_appendFile(name, m_cdrom);
|
||
|
}
|
||
|
SetStatus(hwndDlg, m_cdrom);
|
||
|
}
|
||
|
break;
|
||
|
}
|
||
|
Sleep(100);
|
||
|
MSG msg;
|
||
|
while (PeekMessage(&msg, NULL, WM_KEYFIRST, WM_KEYLAST, PM_REMOVE)); //eat return
|
||
|
}
|
||
|
break;
|
||
|
case IDC_CLEAR:
|
||
|
if (DM_MODE_BURNING != DriveManager_GetDriveMode(m_cdrom))
|
||
|
{
|
||
|
wchar_t title[32] = {0};
|
||
|
if (MessageBox(hwndDlg, WASABI_API_LNGSTRINGW(IDS_SURE_YOU_WANT_TO_CLEAR_BURNING_LIST),
|
||
|
WASABI_API_LNGSTRINGW_BUF(IDS_CONFIRMATION,title,32), MB_YESNO) != IDYES)
|
||
|
break;
|
||
|
emptyRecordList(&itemCache[LETTERTOINDEX(m_cdrom)]);
|
||
|
SetStatus(hwndDlg, m_cdrom);
|
||
|
refreshList();
|
||
|
}
|
||
|
break;
|
||
|
case IDC_CANCEL_BURN:
|
||
|
case IDC_BURN:
|
||
|
{
|
||
|
HWND h;
|
||
|
if (NULL != (h = cdburn_FindBurningHWND(m_cdrom)))
|
||
|
{
|
||
|
PostMessage(h, WM_COMMAND, MAKEWPARAM(IDCANCEL, BN_CLICKED), 0);
|
||
|
EnableWindow(GetDlgItem(hwndDlg, IDC_BURN), FALSE);
|
||
|
EnableWindow(GetDlgItem(hwndDlg, IDC_CANCEL_BURN), FALSE);
|
||
|
|
||
|
}
|
||
|
else doBurnDialog(hwndDlg);
|
||
|
}
|
||
|
break;
|
||
|
case IDC_BTN_SHOWINFO:
|
||
|
switch(HIWORD(wParam))
|
||
|
{
|
||
|
case BN_CLICKED: SendMessageW(GetParent(hwndDlg), WM_COMMAND, wParam, lParam); break;
|
||
|
}
|
||
|
break;
|
||
|
}
|
||
|
break;
|
||
|
|
||
|
case WM_CONTEXTMENU:
|
||
|
{
|
||
|
if (DM_MODE_BURNING == DriveManager_GetDriveMode(m_cdrom) || m_statuslist.GetCount() == 0) return 0;
|
||
|
|
||
|
HMENU menu = GetSubMenu(g_context_menus, 4);
|
||
|
|
||
|
POINT pt = {GET_X_LPARAM(lParam), GET_Y_LPARAM(lParam)};
|
||
|
|
||
|
if (pt.x == -1 || pt.y == -1) // x and y are -1 if the user invoked a shift-f10 popup menu
|
||
|
{
|
||
|
RECT itemRect = {0};
|
||
|
int selected = m_statuslist.GetNextSelected();
|
||
|
if (selected != -1) // if something is selected we'll drop the menu from there
|
||
|
{
|
||
|
m_statuslist.GetItemRect(selected, &itemRect);
|
||
|
ClientToScreen(m_statuslist.getwnd(), (POINT *)&itemRect);
|
||
|
}
|
||
|
else // otherwise we'll drop it from the top-left corner of the listview, adjusting for the header location
|
||
|
{
|
||
|
GetWindowRect(m_statuslist.getwnd(), &itemRect);
|
||
|
|
||
|
HWND hHeader = (HWND)SNDMSG((HWND)wParam, LVM_GETHEADER, 0, 0L);
|
||
|
RECT headerRect;
|
||
|
if ((WS_VISIBLE & GetWindowLongPtr(hHeader, GWL_STYLE)) && GetWindowRect(hHeader, &headerRect))
|
||
|
{
|
||
|
itemRect.top += (headerRect.bottom - headerRect.top);
|
||
|
}
|
||
|
}
|
||
|
pt.x = itemRect.left;
|
||
|
pt.y = itemRect.top;
|
||
|
}
|
||
|
|
||
|
HWND hHeader = (HWND)SNDMSG((HWND)wParam, LVM_GETHEADER, 0, 0L);
|
||
|
RECT headerRect;
|
||
|
if (0 == (WS_VISIBLE & GetWindowLongPtr(hHeader, GWL_STYLE)) || FALSE == GetWindowRect(hHeader, &headerRect))
|
||
|
{
|
||
|
SetRectEmpty(&headerRect);
|
||
|
}
|
||
|
|
||
|
if (FALSE != PtInRect(&headerRect, pt))
|
||
|
{
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
int r = Menu_TrackPopup(plugin.hwndLibraryParent, menu,
|
||
|
TPM_RETURNCMD | TPM_RIGHTBUTTON | TPM_LEFTBUTTON | TPM_NONOTIFY,
|
||
|
pt.x, pt.y, hwndDlg, NULL);
|
||
|
switch (r)
|
||
|
{
|
||
|
case ID_BURNCONTEXTMENU_PLAYSELECTEDITEMS:
|
||
|
playSelectedItems(hwndDlg, 0);
|
||
|
break;
|
||
|
case ID_BURNCONTEXTMENU_ENQUEUESELECTEDITEMS:
|
||
|
playSelectedItems(hwndDlg, 1);
|
||
|
break;
|
||
|
case ID_BURNCONTEXTMENU_SELECTALL:
|
||
|
selectAll();
|
||
|
break;
|
||
|
case ID_BURNCONTEXTMENU_REMOVESELECTEDITEMS:
|
||
|
if (DM_MODE_BURNING != DriveManager_GetDriveMode(m_cdrom)) deleteSelectedItems(hwndDlg, m_cdrom);
|
||
|
break;
|
||
|
case ID_BURNCONTEXTMENU_BURN:
|
||
|
doBurnDialog(hwndDlg);
|
||
|
break;
|
||
|
case ID_BURNCONTEXTMENU_MOVESELECTEDITEMSUP:
|
||
|
moveSelItemsUp();
|
||
|
break;
|
||
|
case ID_BURNCONTEXTMENU_MOVESELECTEDITEMSDOWN:
|
||
|
moveSelItemsDown();
|
||
|
break;
|
||
|
}
|
||
|
Sleep(100);
|
||
|
MSG msg;
|
||
|
while (PeekMessage(&msg, NULL, WM_KEYFIRST, WM_KEYLAST, PM_REMOVE)); //eat return
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
case WM_NOTIFY:
|
||
|
{
|
||
|
LPNMHDR l = (LPNMHDR)lParam;
|
||
|
if (l->idFrom == IDC_LIST2)
|
||
|
{
|
||
|
if (l->code == LVN_KEYDOWN)
|
||
|
{
|
||
|
LPNMLVKEYDOWN pnkd = (LPNMLVKEYDOWN) lParam;
|
||
|
switch (pnkd->wVKey)
|
||
|
{
|
||
|
case 38: //up
|
||
|
if (GetAsyncKeyState(VK_LMENU))
|
||
|
{
|
||
|
moveSelItemsUp();
|
||
|
return 1;
|
||
|
}
|
||
|
break;
|
||
|
case 40: //down
|
||
|
if (GetAsyncKeyState(VK_LMENU))
|
||
|
{
|
||
|
moveSelItemsDown();
|
||
|
return 1;
|
||
|
}
|
||
|
break;
|
||
|
case 46: //delete
|
||
|
if (DM_MODE_BURNING != DriveManager_GetDriveMode(m_cdrom)) deleteSelectedItems(hwndDlg, m_cdrom);
|
||
|
break;
|
||
|
case 'A':
|
||
|
if (GetAsyncKeyState(VK_CONTROL))
|
||
|
selectAll();
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
else if (l->code == NM_DBLCLK)
|
||
|
{
|
||
|
if (DM_MODE_BURNING == DriveManager_GetDriveMode(m_cdrom)) return 0;
|
||
|
playSelectedItems(hwndDlg, (!!g_config->ReadInt(L"enqueuedef", 0)) ^(!!(GetAsyncKeyState(VK_SHIFT)&0x8000)));
|
||
|
}
|
||
|
else if (l->code == NM_RETURN)
|
||
|
{
|
||
|
if (DM_MODE_BURNING == DriveManager_GetDriveMode(m_cdrom)) return 0;
|
||
|
playSelectedItems(hwndDlg, 0 ^(!!(GetAsyncKeyState(VK_SHIFT)&0x8000)));
|
||
|
}
|
||
|
else if (l->code == LVN_GETDISPINFO)
|
||
|
{
|
||
|
NMLVDISPINFO *lpdi = (NMLVDISPINFO*) lParam;
|
||
|
int item = lpdi->item.iItem;
|
||
|
int idx = LETTERTOINDEX(m_cdrom);
|
||
|
if (item < 0 || item >= itemCache[idx].Size) return 0;
|
||
|
|
||
|
itemRecordW *thisitem = itemCache[idx].Items + item;
|
||
|
|
||
|
if (lpdi->item.mask & (LVIF_TEXT | /*LVIF_IMAGE*/0)) // we can always do images too :)
|
||
|
{
|
||
|
if (lpdi->item.mask & LVIF_TEXT)
|
||
|
{
|
||
|
wchar_t tmpbuf[128] = {0};
|
||
|
wchar_t *nameptr = 0;
|
||
|
switch (lpdi->item.iSubItem)
|
||
|
{
|
||
|
case 0:
|
||
|
//track #
|
||
|
StringCchPrintfW(tmpbuf, 128, L"%d", item + 1);
|
||
|
nameptr = tmpbuf;
|
||
|
break;
|
||
|
case 1:
|
||
|
//title
|
||
|
lstrcpynW(tmpbuf, thisitem->title, 128);
|
||
|
nameptr = tmpbuf;
|
||
|
break;
|
||
|
case 2:
|
||
|
//length
|
||
|
StringCchPrintfW(tmpbuf, 128, L"%01d:%02d", thisitem->length / 60, thisitem->length % 60);
|
||
|
nameptr = tmpbuf;
|
||
|
break;
|
||
|
case 3:
|
||
|
DWORD state = (DWORD) SendMessage(m_burning_other_wnd, WM_BURNGETITEMSTATUS, BURNSTATUS_STATE, (LPARAM)item);
|
||
|
switch (state)
|
||
|
{
|
||
|
case BURNERITEM_BURNING:
|
||
|
case BURNERITEM_DECODING:
|
||
|
StringCchPrintfW(tmpbuf, 128, L"%s (%d%%)",
|
||
|
WASABI_API_LNGSTRINGW((BURNERITEM_BURNING == state) ? IDS_BURNING_ : IDS_PREPARING),
|
||
|
(DWORD)SendMessage(m_burning_other_wnd, WM_BURNGETITEMSTATUS, BURNSTATUS_PROGRESS, (LPARAM)item));
|
||
|
nameptr = tmpbuf;
|
||
|
break;
|
||
|
case BURNERITEM_SUCCESS: break;
|
||
|
case BURNERITEM_BURNED:
|
||
|
nameptr = WASABI_API_LNGSTRINGW_BUF(IDS_FINISHED,tmpbuf,128);
|
||
|
break;
|
||
|
case BURNERITEM_DECODED:
|
||
|
nameptr = WASABI_API_LNGSTRINGW_BUF(IDS_PREPARED,tmpbuf,128);
|
||
|
break;
|
||
|
case BURNERITEM_SKIPPED:
|
||
|
nameptr = WASABI_API_LNGSTRINGW_BUF(IDS_SKIPPED,tmpbuf,128);
|
||
|
break;
|
||
|
case BURNERITEM_READY:
|
||
|
nameptr = WASABI_API_LNGSTRINGW_BUF(IDS_SCHEDULED,tmpbuf,128);
|
||
|
break;
|
||
|
case BURNERITEM_LICENSING:
|
||
|
nameptr = WASABI_API_LNGSTRINGW_BUF(IDS_CHECKING_LICENSE,tmpbuf,128);
|
||
|
break;
|
||
|
case BURNERITEM_LICENSED:
|
||
|
nameptr = WASABI_API_LNGSTRINGW_BUF(IDS_LICENSED,tmpbuf,128);
|
||
|
break;
|
||
|
case BURNERITEM_ABORTED:
|
||
|
nameptr = WASABI_API_LNGSTRINGW_BUF(IDS_CANCELLED,tmpbuf,128);
|
||
|
break;
|
||
|
case BURNERITEM_FAILED:
|
||
|
nameptr = WASABI_API_LNGSTRINGW_BUF(IDS_FAILED,tmpbuf,128);
|
||
|
break;
|
||
|
case BURNERITEM_CANCELING:
|
||
|
nameptr = WASABI_API_LNGSTRINGW_BUF(IDS_CANCELLING,tmpbuf,128);
|
||
|
break;
|
||
|
case BURNERITEM_BADFILENAME:
|
||
|
nameptr = WASABI_API_LNGSTRINGW_BUF(IDS_BAD_FILENAME,tmpbuf,128);
|
||
|
break;
|
||
|
case BURNERITEM_UNABLEOPENFILE:
|
||
|
nameptr = WASABI_API_LNGSTRINGW_BUF(IDS_UNABLE_TO_OPEN_FILE,tmpbuf,128);
|
||
|
break;
|
||
|
case BURNERITEM_WRITEERROR:
|
||
|
nameptr = WASABI_API_LNGSTRINGW_BUF(IDS_CACHE_WRITE_FAILED,tmpbuf,128);
|
||
|
break;
|
||
|
case BURNERITEM_DECODEERROR:
|
||
|
nameptr = WASABI_API_LNGSTRINGW_BUF(IDS_UNABLE_TO_FIND_DECODER,tmpbuf,128);
|
||
|
break;
|
||
|
case BURNERITEM_ADDSTREAMFAILED:
|
||
|
nameptr = WASABI_API_LNGSTRINGW_BUF(IDS_CANNOT_ADD_TO_THE_DISC,tmpbuf,128);
|
||
|
break;
|
||
|
case BURNERITEM_READSTREAMERROR:
|
||
|
nameptr = WASABI_API_LNGSTRINGW_BUF(IDS_CACHE_READ_FAILED,tmpbuf,128);
|
||
|
break;
|
||
|
default:
|
||
|
nameptr = WASABI_API_LNGSTRINGW_BUF(IDS_UNKNOWN_ERROR,tmpbuf,128);
|
||
|
break;
|
||
|
}
|
||
|
//status
|
||
|
break;
|
||
|
}
|
||
|
if (nameptr)
|
||
|
lstrcpynW(lpdi->item.pszText, nameptr, lpdi->item.cchTextMax);
|
||
|
else
|
||
|
lpdi->item.pszText[0] = 0;
|
||
|
}
|
||
|
}
|
||
|
return 0;
|
||
|
}
|
||
|
else if (l->code == LVN_BEGINDRAG)
|
||
|
{
|
||
|
SetCapture(hwndDlg);
|
||
|
m_dragging = 1;
|
||
|
LPNMLISTVIEW nlv = (LPNMLISTVIEW)lParam;
|
||
|
m_drag_item = nlv->iItem;
|
||
|
}
|
||
|
else if (l->code == LVN_ITEMCHANGED) ListView_OnItemChanged(hwndDlg, (NMLISTVIEW*)l);
|
||
|
}
|
||
|
}
|
||
|
break;
|
||
|
|
||
|
case WM_MOUSEMOVE:
|
||
|
if (m_dragging)
|
||
|
{
|
||
|
POINT p;
|
||
|
p.x = GET_X_LPARAM(lParam);
|
||
|
p.y = GET_Y_LPARAM(lParam);
|
||
|
ClientToScreen(hwndDlg, &p);
|
||
|
ScreenToClient(m_statuslist.getwnd(), &p);
|
||
|
int i = m_statuslist.FindItemByPoint(p.x, p.y);
|
||
|
if (i != -1 && i != m_drag_item)
|
||
|
{
|
||
|
if (i > m_drag_item)
|
||
|
{
|
||
|
for (int j = 0;j < (i - m_drag_item);j++)
|
||
|
moveSelItemsDown();
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
for (int j = 0;j < (m_drag_item - i);j++)
|
||
|
moveSelItemsUp();
|
||
|
}
|
||
|
m_drag_item = i;
|
||
|
}
|
||
|
}
|
||
|
break;
|
||
|
|
||
|
case WM_LBUTTONUP:
|
||
|
if (GetCapture() == hwndDlg)
|
||
|
{
|
||
|
ReleaseCapture();
|
||
|
m_dragging = 0;
|
||
|
}
|
||
|
break;
|
||
|
|
||
|
case WM_DESTROY:
|
||
|
if (m_statuslist.getwnd())
|
||
|
{
|
||
|
g_view_metaconf->WriteInt(L"col_track", m_statuslist.GetColumnWidth(0));
|
||
|
g_view_metaconf->WriteInt(L"col_title", m_statuslist.GetColumnWidth(1));
|
||
|
g_view_metaconf->WriteInt(L"col_len", m_statuslist.GetColumnWidth(2));
|
||
|
g_view_metaconf->WriteInt(L"col_status", m_statuslist.GetColumnWidth(3));
|
||
|
}
|
||
|
|
||
|
if (m_burning_other_wnd && IsWindow(m_burning_other_wnd))
|
||
|
{
|
||
|
PostMessage(m_burning_other_wnd, WM_BURNUPDATEOWNER, 0, (LPARAM)prevWnd);
|
||
|
prevWnd = NULL;
|
||
|
}
|
||
|
m_hwndstatus = 0;
|
||
|
|
||
|
if (hPLFont)
|
||
|
{
|
||
|
DeleteObject(hPLFont);
|
||
|
hPLFont = NULL;
|
||
|
}
|
||
|
|
||
|
{
|
||
|
HANDLE hPrev = (HANDLE) SendDlgItemMessage(hwndDlg,IDC_LOGO,STM_SETIMAGE,IMAGE_BITMAP, 0L);
|
||
|
if (hPrev) DeleteObject(hPrev);
|
||
|
}
|
||
|
return 0;
|
||
|
case WM_ML_CHILDIPC:
|
||
|
if (lParam == ML_CHILDIPC_DROPITEM && wParam)
|
||
|
{
|
||
|
mlDropItemStruct *dis = (mlDropItemStruct *)wParam;
|
||
|
if (DM_MODE_BURNING == DriveManager_GetDriveMode(m_cdrom))
|
||
|
{
|
||
|
dis->result = -1;
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
if (dis->type != ML_TYPE_ITEMRECORDLISTW && dis->type != ML_TYPE_ITEMRECORDLIST &&
|
||
|
dis->type != ML_TYPE_FILENAMES && dis->type != ML_TYPE_FILENAMESW)
|
||
|
{
|
||
|
dis->result = -1;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
if (dis->data)
|
||
|
{
|
||
|
dis->result = 1;
|
||
|
if (dis->type == ML_TYPE_ITEMRECORDLIST)
|
||
|
{
|
||
|
itemRecordList *obj = (itemRecordList *)dis->data;
|
||
|
cdburn_appendItemRecord(obj, m_cdrom);
|
||
|
}
|
||
|
else if (dis->type == ML_TYPE_ITEMRECORDLISTW)
|
||
|
{
|
||
|
itemRecordListW *obj = (itemRecordListW *)dis->data;
|
||
|
cdburn_appendItemRecord(obj, m_cdrom);
|
||
|
}
|
||
|
else if (dis->type == ML_TYPE_FILENAMES) // playlist
|
||
|
{
|
||
|
char *p = (char*)dis->data;
|
||
|
while (p && *p)
|
||
|
{
|
||
|
cdburn_appendFile(p, m_cdrom);
|
||
|
p += strlen(p) + 1;
|
||
|
}
|
||
|
}
|
||
|
else if (dis->type == ML_TYPE_FILENAMESW)
|
||
|
{
|
||
|
wchar_t *p = (wchar_t*)dis->data;
|
||
|
while (p && *p)
|
||
|
{
|
||
|
cdburn_appendFile(p, m_cdrom);
|
||
|
p += wcslen(p) + 1;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
return 0;
|
||
|
|
||
|
case WM_DROPFILES:
|
||
|
{
|
||
|
char temp[2048] = {0};
|
||
|
HDROP h = (HDROP) wParam;
|
||
|
int y = DragQueryFileA(h, 0xffffffff, temp, sizeof(temp));
|
||
|
|
||
|
if (DM_MODE_BURNING == DriveManager_GetDriveMode(m_cdrom))
|
||
|
{
|
||
|
// MessageBoxA(hwndDlg,"Cannot add files while burning","CD Burner",MB_OK);
|
||
|
}
|
||
|
else for (int x = 0; x < y; x ++)
|
||
|
{
|
||
|
DragQueryFileA(h, x, temp, sizeof(temp));
|
||
|
cdburn_appendFile(temp, m_cdrom);
|
||
|
}
|
||
|
|
||
|
DragFinish(h);
|
||
|
}
|
||
|
return 0;
|
||
|
case WM_PAINT:
|
||
|
{
|
||
|
int tab[] = { IDC_LIST2 | DCW_SUNKENBORDER, IDC_LOGO | DCW_SUNKENBORDER};
|
||
|
dialogSkinner.Draw(hwndDlg, tab, 2);
|
||
|
}
|
||
|
return 0;
|
||
|
case WM_ERASEBKGND: return 1; //handled by WADlg_DrawChildWindowBorders in WM_PAINT
|
||
|
case WM_QUERYFILEINFO: Window_OnQueryInfo(hwndDlg); break;
|
||
|
case WM_EX_OPCOMPLETED: Window_OnOperationCompleted(hwndDlg, (DM_NOTIFY_PARAM*)lParam);
|
||
|
}
|
||
|
return FALSE;
|
||
|
}
|
||
|
|
||
|
static HWND BurnAddStatus_wnd;
|
||
|
|
||
|
static BOOL CALLBACK BurnAddStatus_proc(HWND hwndDlg, UINT uMsg, WPARAM wParam, LPARAM lParam)
|
||
|
{
|
||
|
if (uMsg == WM_CLOSE)
|
||
|
DestroyWindow(hwndDlg);
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
void BurnAddStatus_Create(int num)
|
||
|
{
|
||
|
if (BurnAddStatus_wnd && IsWindow(BurnAddStatus_wnd)) DestroyWindow(BurnAddStatus_wnd);
|
||
|
BurnAddStatus_wnd = WASABI_API_CREATEDIALOGW(IDD_BURN_ADD_STATUS, plugin.hwndLibraryParent, BurnAddStatus_proc);
|
||
|
if (!BurnAddStatus_wnd) return ;
|
||
|
|
||
|
SetTimer(BurnAddStatus_wnd, 1, 100, NULL);
|
||
|
SendDlgItemMessage(BurnAddStatus_wnd, IDC_PROGRESS1, PBM_SETRANGE, 0, MAKELPARAM(0, num));
|
||
|
|
||
|
unsigned int start_t = GetTickCount();
|
||
|
if (start_t >= 0xffffff00) start_t = 0;
|
||
|
|
||
|
MSG msg;
|
||
|
while (GetTickCount() < start_t + 100 && GetMessage(&msg, NULL, 0, 0))
|
||
|
{
|
||
|
TranslateMessage(&msg);
|
||
|
DispatchMessage(&msg);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void BurnAddStatus_Step(itemRecordListW *items)
|
||
|
{
|
||
|
if (!BurnAddStatus_wnd || !items || items && !items->Size || items && !items->Items) return ;
|
||
|
SendDlgItemMessage(BurnAddStatus_wnd, IDC_PROGRESS1, PBM_DELTAPOS, 1, 0);
|
||
|
|
||
|
int l = 0;
|
||
|
for (int i = 0;i < items->Size;i++)
|
||
|
{
|
||
|
l += items->Items[i].length;
|
||
|
}
|
||
|
wchar_t buf[512] = {0};
|
||
|
StringCchPrintf(buf, 512, WASABI_API_LNGSTRINGW(IDS_ADDING_TRACKS_TO_BURNER_TOTAL_LENGTH_X), l / 60, l % 60);
|
||
|
SetDlgItemText(BurnAddStatus_wnd, IDC_STAT, buf);
|
||
|
}
|
||
|
|
||
|
void BurnAddStatus_Done()
|
||
|
{
|
||
|
if (!BurnAddStatus_wnd) return ;
|
||
|
unsigned int start_t = GetTickCount();
|
||
|
if (start_t >= 0xffffff00) start_t = 0;
|
||
|
|
||
|
MSG msg;
|
||
|
while (GetTickCount() < start_t + 1000 && IsWindow(BurnAddStatus_wnd) && GetMessage(&msg, NULL, 0, 0))
|
||
|
{
|
||
|
TranslateMessage(&msg);
|
||
|
DispatchMessage(&msg);
|
||
|
}
|
||
|
|
||
|
DestroyWindow(BurnAddStatus_wnd);
|
||
|
BurnAddStatus_wnd = 0;
|
||
|
}
|
||
|
|
||
|
static bool cdrFound(char letter)
|
||
|
{
|
||
|
wchar_t name[]= L"cda://X.cda";
|
||
|
wchar_t info[16] = L"";
|
||
|
name[6] = letter;
|
||
|
getFileInfoW(name, L"cdtype", info, sizeof(info)/sizeof(wchar_t));
|
||
|
return !lstrcmpW(info, L"CDR") || !lstrcmpW(info, L"CDRW");
|
||
|
}
|
||
|
|
||
|
static char m_burnwait_letter;
|
||
|
static BOOL CALLBACK BurnWaitProc(HWND hwndDlg, UINT uMsg, WPARAM wParam, LPARAM lParam)
|
||
|
{
|
||
|
switch (uMsg)
|
||
|
{
|
||
|
case WM_INITDIALOG:
|
||
|
SetTimer(hwndDlg, 1, 2000, NULL);
|
||
|
{
|
||
|
wchar_t buf[512] = {0};
|
||
|
StringCchPrintf(buf, 512, WASABI_API_LNGSTRINGW(IDS_PLEASE_INSERT_BLANK_RECORDABLE_CD), toupper(m_burnwait_letter));
|
||
|
SetDlgItemText(hwndDlg, IDC_TEXT, buf);
|
||
|
}
|
||
|
return 0;
|
||
|
case WM_COMMAND:
|
||
|
if (LOWORD(wParam) == IDCANCEL) EndDialog(hwndDlg, 1);
|
||
|
return 0;
|
||
|
case WM_TIMER:
|
||
|
if (cdrFound(m_burnwait_letter)) EndDialog(hwndDlg, 0);
|
||
|
return 0;
|
||
|
}
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
int Burn_WaitForCDR(HWND hwndParent, char driveletter) // returns 0 on CD-R found, 1 on cancel
|
||
|
{
|
||
|
CHAR cMode;
|
||
|
if (!driveletter) return 1;
|
||
|
cMode = DriveManager_GetDriveMode(driveletter);
|
||
|
if (DM_MODE_BURNING == cMode || DM_MODE_RIPPING == cMode)
|
||
|
{
|
||
|
return 1; // if burning or ripping, don't fuck with it
|
||
|
}
|
||
|
|
||
|
if (cdrFound(driveletter)) return 0;
|
||
|
if (m_burnwait_letter) return 1;
|
||
|
|
||
|
m_burnwait_letter = (char)toupper(driveletter);
|
||
|
int x = (int)(INT_PTR)WASABI_API_DIALOGBOXW(IDD_WAITFORCDR, hwndParent, BurnWaitProc);
|
||
|
m_burnwait_letter = 0;
|
||
|
return x;
|
||
|
}
|