* layout.itemR now has icon_only option to show only icon in e.g. enums buttons, for uv editor header. * Automatic key shortcuts in menus now show the shortcut even if operator properties don't match. Not sure this will work well everywhere, but seems to be working ok for now. * Open recent now show shorter filenames instead of the whole file path. * Tweak object Duplicate menu item.
427 lines
8.6 KiB
C
427 lines
8.6 KiB
C
/*
|
|
* $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.
|
|
*
|
|
* Contributor(s): none yet.
|
|
*
|
|
* ***** END GPL LICENSE BLOCK *****
|
|
*/
|
|
|
|
#include <string.h>
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
|
|
#include <sys/types.h>
|
|
#include <sys/stat.h>
|
|
#include <fcntl.h>
|
|
|
|
#include <errno.h>
|
|
|
|
#include "zlib.h"
|
|
|
|
#ifdef WIN32
|
|
#include <io.h>
|
|
#include "BLI_winstuff.h"
|
|
#else
|
|
#include <unistd.h> // for read close
|
|
#include <sys/param.h>
|
|
#endif
|
|
|
|
|
|
#include "BLI_blenlib.h"
|
|
#include "BLI_storage.h"
|
|
#include "BLI_fileops.h"
|
|
#include "BLI_callbacks.h"
|
|
|
|
#include "BKE_utildefines.h"
|
|
|
|
#include "BLO_sys_types.h" // for intptr_t support
|
|
|
|
/* implementations: */
|
|
char *first_slash(char *string) {
|
|
char *ffslash, *fbslash;
|
|
|
|
ffslash= strchr(string, '/');
|
|
fbslash= strchr(string, '\\');
|
|
|
|
if (!ffslash) return fbslash;
|
|
else if (!fbslash) return ffslash;
|
|
|
|
if ((intptr_t)ffslash < (intptr_t)fbslash) return ffslash;
|
|
else return fbslash;
|
|
}
|
|
|
|
char *BLI_last_slash(const char *string) {
|
|
char *lfslash, *lbslash;
|
|
|
|
lfslash= strrchr(string, '/');
|
|
lbslash= strrchr(string, '\\');
|
|
|
|
if (!lfslash) return lbslash;
|
|
else if (!lbslash) return lfslash;
|
|
|
|
if ((intptr_t)lfslash < (intptr_t)lbslash) return lbslash;
|
|
else return lfslash;
|
|
}
|
|
|
|
static const char *last_slash_len(const char *string, int len) {
|
|
int a;
|
|
|
|
for(a=len-1; a>=0; a--)
|
|
if(string[a] == '/' || string[a] == '\\')
|
|
return &string[a];
|
|
|
|
return NULL;
|
|
}
|
|
|
|
const char *BLI_short_filename(const char *string) {
|
|
const char *ls, *lls;
|
|
|
|
ls= last_slash_len(string, strlen(string));
|
|
if(!ls)
|
|
return string;
|
|
|
|
lls= last_slash_len(string, ls-string);
|
|
|
|
if(lls)
|
|
return lls+1;
|
|
else
|
|
return ls+1;
|
|
}
|
|
|
|
/* adds a slash if there isnt one there alredy */
|
|
int BLI_add_slash(char *string) {
|
|
int len = strlen(string);
|
|
#ifdef WIN32
|
|
if (len==0 || string[len-1]!='\\') {
|
|
string[len] = '\\';
|
|
string[len+1] = '\0';
|
|
return len+1;
|
|
}
|
|
#else
|
|
if (len==0 || string[len-1]!='/') {
|
|
string[len] = '/';
|
|
string[len+1] = '\0';
|
|
return len+1;
|
|
}
|
|
#endif
|
|
return len;
|
|
}
|
|
|
|
/* removes a slash if there is one */
|
|
void BLI_del_slash(char *string) {
|
|
int len = strlen(string);
|
|
while (len) {
|
|
#ifdef WIN32
|
|
if (string[len-1]=='\\') {
|
|
#else
|
|
if (string[len-1]=='/') {
|
|
#endif
|
|
string[len-1] = '\0';
|
|
len--;
|
|
} else {
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
/* gzip the file in from and write it to "to".
|
|
return -1 if zlib fails, -2 if the originating file does not exist
|
|
note: will remove the "from" file
|
|
*/
|
|
int BLI_gzip(char *from, char *to) {
|
|
char buffer[10240];
|
|
int file;
|
|
int readsize = 0;
|
|
|
|
gzFile gzfile = gzopen(to,"wb");
|
|
if (NULL == gzfile) return -1;
|
|
|
|
file = open(from,O_BINARY|O_RDONLY);
|
|
|
|
if ( -1 == file ) return -2;
|
|
|
|
while ( 1 )
|
|
{
|
|
readsize = read(file, buffer, 10240);
|
|
|
|
if (readsize <= 0) break;
|
|
|
|
gzwrite(gzfile,buffer,readsize);
|
|
}
|
|
|
|
gzclose(gzfile);
|
|
close(file);
|
|
|
|
remove(from);
|
|
|
|
return 0;
|
|
}
|
|
|
|
/* return 1 when file can be written */
|
|
int BLI_is_writable(char *filename)
|
|
{
|
|
int file;
|
|
|
|
/* first try to open without creating */
|
|
file = open(filename, O_BINARY | O_RDWR, 0666);
|
|
|
|
if (file < 0) {
|
|
/* now try to open and create. a test without actually
|
|
* creating a file would be nice, but how? */
|
|
file = open(filename, O_BINARY | O_RDWR | O_CREAT, 0666);
|
|
|
|
if(file < 0) {
|
|
return 0;
|
|
}
|
|
else {
|
|
/* success, delete the file we create */
|
|
close(file);
|
|
BLI_delete(filename, 0, 0);
|
|
return 1;
|
|
}
|
|
}
|
|
else {
|
|
close(file);
|
|
return 1;
|
|
}
|
|
}
|
|
|
|
int BLI_touch(const char *file)
|
|
{
|
|
FILE *f = fopen(file,"r+b");
|
|
if (f != NULL) {
|
|
char c = getc(f);
|
|
rewind(f);
|
|
putc(c,f);
|
|
} else {
|
|
f = fopen(file,"wb");
|
|
}
|
|
if (f) {
|
|
fclose(f);
|
|
return 1;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
#ifdef WIN32
|
|
|
|
static char str[MAXPATHLEN+12];
|
|
|
|
int BLI_delete(char *file, int dir, int recursive) {
|
|
int err;
|
|
|
|
if (recursive) {
|
|
callLocalErrorCallBack("Recursive delete is unsupported on Windows");
|
|
err= 1;
|
|
} else if (dir) {
|
|
err= !RemoveDirectory(file);
|
|
if (err) printf ("Unable to remove directory");
|
|
} else {
|
|
err= !DeleteFile(file);
|
|
if (err) callLocalErrorCallBack("Unable to delete file");
|
|
}
|
|
|
|
return err;
|
|
}
|
|
|
|
int BLI_move(char *file, char *to) {
|
|
int err;
|
|
|
|
// windows doesn't support moveing to a directory
|
|
// it has to be 'mv filename filename' and not
|
|
// 'mv filename destdir'
|
|
|
|
strcpy(str, to);
|
|
// points 'to' to a directory ?
|
|
if (BLI_last_slash(str) == (str + strlen(str) - 1)) {
|
|
if (BLI_last_slash(file) != NULL) {
|
|
strcat(str, BLI_last_slash(file) + 1);
|
|
}
|
|
}
|
|
|
|
err= !MoveFile(file, str);
|
|
if (err) {
|
|
callLocalErrorCallBack("Unable to move file");
|
|
printf(" Move from '%s' to '%s' failed\n", file, str);
|
|
}
|
|
|
|
return err;
|
|
}
|
|
|
|
|
|
int BLI_copy_fileops(char *file, char *to) {
|
|
int err;
|
|
|
|
// windows doesn't support copying to a directory
|
|
// it has to be 'cp filename filename' and not
|
|
// 'cp filename destdir'
|
|
|
|
strcpy(str, to);
|
|
// points 'to' to a directory ?
|
|
if (BLI_last_slash(str) == (str + strlen(str) - 1)) {
|
|
if (BLI_last_slash(file) != NULL) {
|
|
strcat(str, BLI_last_slash(file) + 1);
|
|
}
|
|
}
|
|
|
|
err= !CopyFile(file,str,FALSE);
|
|
|
|
if (err) {
|
|
callLocalErrorCallBack("Unable to copy file!");
|
|
printf(" Copy from '%s' to '%s' failed\n", file, str);
|
|
}
|
|
|
|
return err;
|
|
}
|
|
|
|
int BLI_link(char *file, char *to) {
|
|
callLocalErrorCallBack("Linking files is unsupported on Windows");
|
|
|
|
return 1;
|
|
}
|
|
|
|
int BLI_exists(char *file) {
|
|
return (GetFileAttributes(file) != 0xFFFFFFFF);
|
|
}
|
|
|
|
void BLI_recurdir_fileops(char *dirname) {
|
|
char *lslash;
|
|
char tmp[MAXPATHLEN];
|
|
|
|
// First remove possible slash at the end of the dirname.
|
|
// This routine otherwise tries to create
|
|
// blah1/blah2/ (with slash) after creating
|
|
// blah1/blah2 (without slash)
|
|
|
|
strcpy(tmp, dirname);
|
|
lslash= BLI_last_slash(tmp);
|
|
|
|
if (lslash == tmp + strlen(tmp) - 1) {
|
|
*lslash = 0;
|
|
}
|
|
|
|
if (BLI_exists(tmp)) return;
|
|
|
|
lslash= BLI_last_slash(tmp);
|
|
if (lslash) {
|
|
/* Split about the last slash and recurse */
|
|
*lslash = 0;
|
|
BLI_recurdir_fileops(tmp);
|
|
}
|
|
|
|
if(dirname[0]) /* patch, this recursive loop tries to create a nameless directory */
|
|
if (!CreateDirectory(dirname, NULL))
|
|
callLocalErrorCallBack("Unable to create directory\n");
|
|
}
|
|
|
|
int BLI_rename(char *from, char *to) {
|
|
if (!BLI_exists(from)) return 0;
|
|
|
|
/* make sure the filenames are different (case insensitive) before removing */
|
|
if (BLI_exists(to) && BLI_strcasecmp(from, to))
|
|
if(BLI_delete(to, 0, 0)) return 1;
|
|
|
|
return rename(from, to);
|
|
}
|
|
|
|
#else /* The sane UNIX world */
|
|
|
|
/*
|
|
* but the sane UNIX world is tied to the interface, and the system
|
|
* timer, and... We implement a callback mechanism. The system will
|
|
* have to initialise the callback before the functions will work!
|
|
* */
|
|
static char str[MAXPATHLEN+12];
|
|
|
|
int BLI_delete(char *file, int dir, int recursive)
|
|
{
|
|
if(strchr(file, '"')) {
|
|
printf("Error: not deleted file %s because of quote!\n", file);
|
|
}
|
|
else {
|
|
if (recursive) {
|
|
sprintf(str, "/bin/rm -rf \"%s\"", file);
|
|
return system(str);
|
|
}
|
|
else if (dir) {
|
|
sprintf(str, "/bin/rmdir \"%s\"", file);
|
|
return system(str);
|
|
}
|
|
else {
|
|
return remove(file); //sprintf(str, "/bin/rm -f \"%s\"", file);
|
|
}
|
|
}
|
|
return -1;
|
|
}
|
|
|
|
int BLI_move(char *file, char *to) {
|
|
sprintf(str, "/bin/mv -f \"%s\" \"%s\"", file, to);
|
|
|
|
return system(str);
|
|
}
|
|
|
|
int BLI_copy_fileops(char *file, char *to) {
|
|
sprintf(str, "/bin/cp -rf \"%s\" \"%s\"", file, to);
|
|
|
|
return system(str);
|
|
}
|
|
|
|
int BLI_link(char *file, char *to) {
|
|
sprintf(str, "/bin/ln -f \"%s\" \"%s\"", file, to);
|
|
|
|
return system(str);
|
|
}
|
|
|
|
int BLI_exists(char *file) {
|
|
return BLI_exist(file);
|
|
}
|
|
|
|
void BLI_recurdir_fileops(char *dirname) {
|
|
char *lslash;
|
|
char tmp[MAXPATHLEN];
|
|
|
|
if (BLI_exists(dirname)) return;
|
|
|
|
strcpy(tmp, dirname);
|
|
|
|
lslash= BLI_last_slash(tmp);
|
|
if (lslash) {
|
|
/* Split about the last slash and recurse */
|
|
*lslash = 0;
|
|
BLI_recurdir_fileops(tmp);
|
|
}
|
|
|
|
mkdir(dirname, 0777);
|
|
}
|
|
|
|
int BLI_rename(char *from, char *to) {
|
|
if (!BLI_exists(from)) return 0;
|
|
|
|
if (BLI_exists(to)) if(BLI_delete(to, 0, 0)) return 1;
|
|
|
|
return rename(from, to);
|
|
}
|
|
|
|
#endif
|