2008-12-18 19:21:30 +00:00
|
|
|
/**
|
|
|
|
* $Id$
|
|
|
|
*
|
|
|
|
* ***** BEGIN GPL LICENSE BLOCK *****
|
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU General Public License
|
|
|
|
* as published by the Free Software Foundation; either version 2
|
|
|
|
* of the License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program; if not, write to the Free Software Foundation,
|
|
|
|
* Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
|
|
|
|
*
|
|
|
|
* The Original Code is Copyright (C) 2001-2002 by NaN Holding BV.
|
|
|
|
* All rights reserved.
|
|
|
|
*
|
|
|
|
* The Original Code is: all of this file.
|
|
|
|
*
|
2009-01-06 14:42:54 +00:00
|
|
|
* Contributor(s): Andrea Weikert (c) 2008 Blender Foundation.
|
2008-12-18 19:21:30 +00:00
|
|
|
*
|
|
|
|
* ***** END GPL LICENSE BLOCK *****
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
2009-01-06 14:42:54 +00:00
|
|
|
#include <stdio.h>
|
2008-12-18 19:21:30 +00:00
|
|
|
#include <math.h>
|
|
|
|
|
|
|
|
#include "MEM_guardedalloc.h"
|
|
|
|
|
|
|
|
#include "BLI_blenlib.h"
|
|
|
|
#include "BLI_linklist.h"
|
|
|
|
#include "BLI_dynstr.h"
|
|
|
|
|
2009-01-06 14:42:54 +00:00
|
|
|
#ifdef WIN32
|
|
|
|
#include <windows.h> /* need to include windows.h so _WIN32_IE is defined */
|
|
|
|
#ifndef _WIN32_IE
|
|
|
|
#define _WIN32_IE 0x0400 /* minimal requirements for SHGetSpecialFolderPath on MINGW MSVC has this defined already */
|
|
|
|
#endif
|
|
|
|
#include <shlobj.h> /* for SHGetSpecialFolderPath, has to be done before BLI_winstuff because 'near' is disabled through BLI_windstuff */
|
|
|
|
#include "BLI_winstuff.h"
|
|
|
|
#endif
|
2008-12-18 19:21:30 +00:00
|
|
|
|
|
|
|
#include "fsmenu.h" /* include ourselves */
|
|
|
|
|
|
|
|
|
|
|
|
/* FSMENU HANDLING */
|
|
|
|
|
|
|
|
/* FSMenuEntry's without paths indicate seperators */
|
|
|
|
typedef struct _FSMenuEntry FSMenuEntry;
|
|
|
|
struct _FSMenuEntry {
|
|
|
|
FSMenuEntry *next;
|
|
|
|
|
|
|
|
char *path;
|
|
|
|
short save;
|
2009-03-10 23:14:41 +00:00
|
|
|
short xs, ys;
|
2008-12-18 19:21:30 +00:00
|
|
|
};
|
|
|
|
|
2009-03-12 19:36:59 +00:00
|
|
|
typedef struct FSMenu
|
|
|
|
{
|
|
|
|
FSMenuEntry *fsmenu_system;
|
|
|
|
FSMenuEntry *fsmenu_bookmarks;
|
|
|
|
FSMenuEntry *fsmenu_recent;
|
|
|
|
|
|
|
|
FSMenuCategory selected_category;
|
|
|
|
int selected_entry;
|
|
|
|
|
|
|
|
} FSMenu;
|
|
|
|
|
|
|
|
static FSMenu *g_fsmenu = NULL;
|
2008-12-18 19:21:30 +00:00
|
|
|
|
2009-03-12 19:36:59 +00:00
|
|
|
struct FSMenu* fsmenu_get(void)
|
|
|
|
{
|
|
|
|
if (!g_fsmenu) {
|
|
|
|
g_fsmenu=MEM_callocN(sizeof(struct FSMenu), "fsmenu");
|
|
|
|
}
|
|
|
|
return g_fsmenu;
|
|
|
|
}
|
2009-03-10 23:14:41 +00:00
|
|
|
|
2009-03-12 19:36:59 +00:00
|
|
|
void fsmenu_select_entry(struct FSMenu* fsmenu, FSMenuCategory category, int index)
|
2009-03-10 23:14:41 +00:00
|
|
|
{
|
2009-03-12 19:36:59 +00:00
|
|
|
fsmenu->selected_category = category;
|
|
|
|
fsmenu->selected_entry = index;
|
2009-03-10 23:14:41 +00:00
|
|
|
}
|
|
|
|
|
2009-03-12 19:36:59 +00:00
|
|
|
int fsmenu_is_selected(struct FSMenu* fsmenu, FSMenuCategory category, int index)
|
2009-03-10 23:14:41 +00:00
|
|
|
{
|
2009-03-12 19:36:59 +00:00
|
|
|
return (category==fsmenu->selected_category) && (index==fsmenu->selected_entry);
|
2009-03-10 23:14:41 +00:00
|
|
|
}
|
|
|
|
|
2009-03-12 19:36:59 +00:00
|
|
|
static FSMenuEntry *fsmenu_get_category(struct FSMenu* fsmenu, FSMenuCategory category)
|
2009-03-10 23:14:41 +00:00
|
|
|
{
|
2009-03-12 19:36:59 +00:00
|
|
|
FSMenuEntry *fsms = NULL;
|
2009-03-10 23:14:41 +00:00
|
|
|
|
|
|
|
switch(category) {
|
|
|
|
case FS_CATEGORY_SYSTEM:
|
2009-03-12 19:36:59 +00:00
|
|
|
fsms = fsmenu->fsmenu_system;
|
2009-03-10 23:14:41 +00:00
|
|
|
break;
|
|
|
|
case FS_CATEGORY_BOOKMARKS:
|
2009-03-12 19:36:59 +00:00
|
|
|
fsms = fsmenu->fsmenu_bookmarks;
|
2009-03-10 23:14:41 +00:00
|
|
|
break;
|
|
|
|
case FS_CATEGORY_RECENT:
|
2009-03-12 19:36:59 +00:00
|
|
|
fsms = fsmenu->fsmenu_recent;
|
2009-03-10 23:14:41 +00:00
|
|
|
break;
|
|
|
|
}
|
2009-03-12 19:36:59 +00:00
|
|
|
return fsms;
|
2009-03-10 23:14:41 +00:00
|
|
|
}
|
|
|
|
|
2009-03-12 19:36:59 +00:00
|
|
|
static void fsmenu_set_category(struct FSMenu* fsmenu, FSMenuCategory category, FSMenuEntry *fsms)
|
2009-03-10 23:14:41 +00:00
|
|
|
{
|
|
|
|
switch(category) {
|
|
|
|
case FS_CATEGORY_SYSTEM:
|
2009-03-12 19:36:59 +00:00
|
|
|
fsmenu->fsmenu_system = fsms;
|
2009-03-10 23:14:41 +00:00
|
|
|
break;
|
|
|
|
case FS_CATEGORY_BOOKMARKS:
|
2009-03-12 19:36:59 +00:00
|
|
|
fsmenu->fsmenu_bookmarks = fsms;
|
2009-03-10 23:14:41 +00:00
|
|
|
break;
|
|
|
|
case FS_CATEGORY_RECENT:
|
2009-03-12 19:36:59 +00:00
|
|
|
fsmenu->fsmenu_recent = fsms;
|
2009-03-10 23:14:41 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-03-12 19:36:59 +00:00
|
|
|
int fsmenu_get_nentries(struct FSMenu* fsmenu, FSMenuCategory category)
|
2008-12-18 19:21:30 +00:00
|
|
|
{
|
|
|
|
FSMenuEntry *fsme;
|
|
|
|
int count= 0;
|
|
|
|
|
2009-03-12 19:36:59 +00:00
|
|
|
for (fsme= fsmenu_get_category(fsmenu, category); fsme; fsme= fsme->next)
|
2008-12-18 19:21:30 +00:00
|
|
|
count++;
|
|
|
|
|
|
|
|
return count;
|
|
|
|
}
|
2009-03-10 23:14:41 +00:00
|
|
|
|
2009-03-12 19:36:59 +00:00
|
|
|
char *fsmenu_get_entry(struct FSMenu* fsmenu, FSMenuCategory category, int idx)
|
2008-12-18 19:21:30 +00:00
|
|
|
{
|
|
|
|
FSMenuEntry *fsme;
|
|
|
|
|
2009-03-12 19:36:59 +00:00
|
|
|
for (fsme= fsmenu_get_category(fsmenu, category); fsme && idx; fsme= fsme->next)
|
2008-12-18 19:21:30 +00:00
|
|
|
idx--;
|
|
|
|
|
2009-03-10 23:14:41 +00:00
|
|
|
return fsme?fsme->path:NULL;
|
2008-12-18 19:21:30 +00:00
|
|
|
}
|
2009-03-10 23:14:41 +00:00
|
|
|
|
2009-03-12 19:36:59 +00:00
|
|
|
void fsmenu_set_pos(struct FSMenu* fsmenu, FSMenuCategory category, int idx, short xs, short ys)
|
2008-12-18 19:21:30 +00:00
|
|
|
{
|
|
|
|
FSMenuEntry *fsme;
|
|
|
|
|
2009-03-12 19:36:59 +00:00
|
|
|
for (fsme= fsmenu_get_category(fsmenu, category); fsme && idx; fsme= fsme->next)
|
2008-12-18 19:21:30 +00:00
|
|
|
idx--;
|
|
|
|
|
2009-03-10 23:14:41 +00:00
|
|
|
if (fsme) {
|
|
|
|
fsme->xs = xs;
|
|
|
|
fsme->ys = ys;
|
|
|
|
}
|
2008-12-18 19:21:30 +00:00
|
|
|
}
|
2009-03-10 23:14:41 +00:00
|
|
|
|
2009-03-12 19:36:59 +00:00
|
|
|
int fsmenu_get_pos (struct FSMenu* fsmenu, FSMenuCategory category, int idx, short* xs, short* ys)
|
2008-12-18 19:21:30 +00:00
|
|
|
{
|
|
|
|
FSMenuEntry *fsme;
|
|
|
|
|
2009-03-12 19:36:59 +00:00
|
|
|
for (fsme= fsmenu_get_category(fsmenu, category); fsme && idx; fsme= fsme->next)
|
2009-03-10 23:14:41 +00:00
|
|
|
idx--;
|
2008-12-18 19:21:30 +00:00
|
|
|
|
2009-03-10 23:14:41 +00:00
|
|
|
if (fsme) {
|
|
|
|
*xs = fsme->xs;
|
|
|
|
*ys = fsme->ys;
|
|
|
|
return 1;
|
|
|
|
}
|
2008-12-18 19:21:30 +00:00
|
|
|
|
2009-03-10 23:14:41 +00:00
|
|
|
return 0;
|
2008-12-18 19:21:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-03-12 19:36:59 +00:00
|
|
|
void fsmenu_insert_entry(struct FSMenu* fsmenu, FSMenuCategory category, char *path, int sorted, short save)
|
2008-12-18 19:21:30 +00:00
|
|
|
{
|
|
|
|
FSMenuEntry *prev;
|
|
|
|
FSMenuEntry *fsme;
|
2009-03-12 19:36:59 +00:00
|
|
|
FSMenuEntry *fsms;
|
2008-12-18 19:21:30 +00:00
|
|
|
|
2009-03-12 19:36:59 +00:00
|
|
|
fsms = fsmenu_get_category(fsmenu, category);
|
|
|
|
prev= fsme= fsms;
|
2008-12-18 19:21:30 +00:00
|
|
|
|
|
|
|
for (; fsme; prev= fsme, fsme= fsme->next) {
|
|
|
|
if (fsme->path) {
|
|
|
|
if (BLI_streq(path, fsme->path)) {
|
|
|
|
return;
|
|
|
|
} else if (sorted && strcmp(path, fsme->path)<0) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// if we're bookmarking this, file should come
|
|
|
|
// before the last separator, only automatically added
|
|
|
|
// current dir go after the last sep.
|
|
|
|
if (save) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fsme= MEM_mallocN(sizeof(*fsme), "fsme");
|
|
|
|
fsme->path= BLI_strdup(path);
|
|
|
|
fsme->save = save;
|
|
|
|
|
|
|
|
if (prev) {
|
|
|
|
fsme->next= prev->next;
|
|
|
|
prev->next= fsme;
|
|
|
|
} else {
|
2009-03-12 19:36:59 +00:00
|
|
|
fsme->next= fsms;
|
|
|
|
fsmenu_set_category(fsmenu, category, fsme);
|
2008-12-18 19:21:30 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-03-12 19:36:59 +00:00
|
|
|
void fsmenu_remove_entry(struct FSMenu* fsmenu, FSMenuCategory category, int idx)
|
2008-12-18 19:21:30 +00:00
|
|
|
{
|
2009-03-10 23:14:41 +00:00
|
|
|
FSMenuEntry *prev= NULL, *fsme= NULL;
|
2009-03-12 19:36:59 +00:00
|
|
|
FSMenuEntry *fsms = fsmenu_get_category(fsmenu, category);
|
2008-12-18 19:21:30 +00:00
|
|
|
|
2009-03-12 19:36:59 +00:00
|
|
|
for (fsme= fsms; fsme && idx; prev= fsme, fsme= fsme->next)
|
2008-12-18 19:21:30 +00:00
|
|
|
idx--;
|
|
|
|
|
|
|
|
if (fsme) {
|
|
|
|
/* you should only be able to remove entries that were
|
|
|
|
not added by default, like windows drives.
|
|
|
|
also separators (where path == NULL) shouldn't be removed */
|
|
|
|
if (fsme->save && fsme->path) {
|
|
|
|
|
|
|
|
/* remove fsme from list */
|
|
|
|
if (prev) {
|
|
|
|
prev->next= fsme->next;
|
|
|
|
} else {
|
2009-03-12 19:36:59 +00:00
|
|
|
fsms= fsme->next;
|
|
|
|
fsmenu_set_category(fsmenu, category, fsms);
|
2008-12-18 19:21:30 +00:00
|
|
|
}
|
|
|
|
/* free entry */
|
|
|
|
MEM_freeN(fsme->path);
|
|
|
|
MEM_freeN(fsme);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-03-12 19:36:59 +00:00
|
|
|
void fsmenu_write_file(struct FSMenu* fsmenu, const char *filename)
|
2008-12-18 19:21:30 +00:00
|
|
|
{
|
2009-03-10 23:14:41 +00:00
|
|
|
FSMenuEntry *fsme= NULL;
|
2009-03-12 19:36:59 +00:00
|
|
|
int count=FSMENU_RECENT_MAX;
|
2008-12-18 19:21:30 +00:00
|
|
|
|
|
|
|
FILE *fp = fopen(filename, "w");
|
|
|
|
if (!fp) return;
|
2009-03-12 19:36:59 +00:00
|
|
|
|
|
|
|
fprintf(fp, "[Bookmarks]\n");
|
|
|
|
for (fsme= fsmenu_get_category(fsmenu, FS_CATEGORY_BOOKMARKS); fsme; fsme= fsme->next) {
|
|
|
|
if (fsme->path && fsme->save) {
|
|
|
|
fprintf(fp, "%s\n", fsme->path);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
fprintf(fp, "[Recent]\n");
|
|
|
|
for (fsme= fsmenu_get_category(fsmenu, FS_CATEGORY_RECENT); fsme && count; fsme= fsme->next, --count) {
|
2008-12-18 19:21:30 +00:00
|
|
|
if (fsme->path && fsme->save) {
|
|
|
|
fprintf(fp, "%s\n", fsme->path);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
fclose(fp);
|
|
|
|
}
|
|
|
|
|
2009-03-12 19:36:59 +00:00
|
|
|
void fsmenu_read_file(struct FSMenu* fsmenu, const char *filename)
|
2009-01-06 14:42:54 +00:00
|
|
|
{
|
|
|
|
char line[256];
|
2009-03-12 19:36:59 +00:00
|
|
|
FSMenuCategory category = FS_CATEGORY_BOOKMARKS;
|
2009-01-06 14:42:54 +00:00
|
|
|
FILE *fp;
|
|
|
|
|
|
|
|
#ifdef WIN32
|
|
|
|
/* Add the drive names to the listing */
|
|
|
|
{
|
|
|
|
__int64 tmp;
|
|
|
|
char folder[256];
|
|
|
|
char tmps[4];
|
|
|
|
int i;
|
|
|
|
|
|
|
|
tmp= GetLogicalDrives();
|
|
|
|
|
|
|
|
for (i=2; i < 26; i++) {
|
|
|
|
if ((tmp>>i) & 1) {
|
|
|
|
tmps[0]='a'+i;
|
|
|
|
tmps[1]=':';
|
|
|
|
tmps[2]='\\';
|
|
|
|
tmps[3]=0;
|
|
|
|
|
2009-03-12 19:36:59 +00:00
|
|
|
fsmenu_insert_entry(fsmenu, FS_CATEGORY_SYSTEM, tmps, 1, 0);
|
2009-01-06 14:42:54 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Adding Desktop and My Documents */
|
|
|
|
SHGetSpecialFolderPath(0, folder, CSIDL_PERSONAL, 0);
|
2009-03-12 19:36:59 +00:00
|
|
|
fsmenu_insert_entry(fsmenu,FS_CATEGORY_BOOKMARKS, folder, 1, 0);
|
2009-01-06 14:42:54 +00:00
|
|
|
SHGetSpecialFolderPath(0, folder, CSIDL_DESKTOPDIRECTORY, 0);
|
2009-03-12 19:36:59 +00:00
|
|
|
fsmenu_insert_entry(fsmenu, FS_CATEGORY_BOOKMARKS, folder, 1, 0);
|
2009-01-06 14:42:54 +00:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2009-02-01 13:52:11 +00:00
|
|
|
fp = fopen(filename, "r");
|
2009-01-06 14:42:54 +00:00
|
|
|
if (!fp) return;
|
|
|
|
|
|
|
|
while ( fgets ( line, 256, fp ) != NULL ) /* read a line */
|
|
|
|
{
|
2009-03-12 19:36:59 +00:00
|
|
|
if (strncmp(line, "[Bookmarks]", 11)==0){
|
|
|
|
category = FS_CATEGORY_BOOKMARKS;
|
|
|
|
} else if (strncmp(line, "[Recent]", 8)==0){
|
|
|
|
category = FS_CATEGORY_RECENT;
|
|
|
|
} else {
|
|
|
|
int len = strlen(line);
|
|
|
|
if (len>0) {
|
|
|
|
if (line[len-1] == '\n') {
|
|
|
|
line[len-1] = '\0';
|
|
|
|
}
|
|
|
|
fsmenu_insert_entry(fsmenu, category, line, 0, 1);
|
2009-01-06 14:42:54 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
fclose(fp);
|
|
|
|
}
|
|
|
|
|
2009-03-12 19:36:59 +00:00
|
|
|
static void fsmenu_free_category(struct FSMenu* fsmenu, FSMenuCategory category)
|
2008-12-18 19:21:30 +00:00
|
|
|
{
|
2009-03-12 19:36:59 +00:00
|
|
|
FSMenuEntry *fsme= fsmenu_get_category(fsmenu, category);
|
2008-12-18 19:21:30 +00:00
|
|
|
|
|
|
|
while (fsme) {
|
|
|
|
FSMenuEntry *n= fsme->next;
|
|
|
|
|
|
|
|
if (fsme->path) MEM_freeN(fsme->path);
|
|
|
|
MEM_freeN(fsme);
|
|
|
|
|
|
|
|
fsme= n;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-03-12 19:36:59 +00:00
|
|
|
void fsmenu_free(struct FSMenu* fsmenu)
|
2009-03-10 23:14:41 +00:00
|
|
|
{
|
2009-03-12 19:36:59 +00:00
|
|
|
fsmenu_free_category(fsmenu, FS_CATEGORY_SYSTEM);
|
|
|
|
fsmenu_free_category(fsmenu, FS_CATEGORY_BOOKMARKS);
|
|
|
|
fsmenu_free_category(fsmenu, FS_CATEGORY_RECENT);
|
|
|
|
MEM_freeN(fsmenu);
|
2009-03-10 23:14:41 +00:00
|
|
|
}
|
2008-12-18 19:21:30 +00:00
|
|
|
|