fltk/src/Fl_File_Chooser2.cxx
Michael R Sweet 88d54cd78b Massive update to use strlcpy() and strlcat() instead of strncpy()
and strncat() in almost all places (there are still a few strncpy's
that need to be used...)

Added configure check for strlcat() and strlcpy().

Added emulation code for strlcat() and strlcpy().


git-svn-id: file:///fltk/svn/fltk/branches/branch-1.1@2239 ea41ed52-d2ee-0310-a9c1-e6b18d33e121
2002-05-16 12:47:44 +00:00

688 lines
17 KiB
C++

//
// "$Id: Fl_File_Chooser2.cxx,v 1.1.2.13 2002/05/16 12:47:43 easysw Exp $"
//
// More Fl_File_Chooser routines.
//
// Copyright 1999-2002 by Michael Sweet.
//
// This library is free software; you can redistribute it and/or
// modify it under the terms of the GNU Library General Public
// License as published by the Free Software Foundation; either
// version 2 of the License, or (at your option) any later version.
//
// This library 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
// Library General Public License for more details.
//
// You should have received a copy of the GNU Library General Public
// License along with this library; if not, write to the Free Software
// Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307
// USA.
//
// Please report all bugs and problems to "fltk-bugs@fltk.org".
//
// Contents:
//
// Fl_File_Chooser::directory() - Set the directory in the file chooser.
// Fl_File_Chooser::count() - Return the number of selected files.
// Fl_File_Chooser::value() - Return a selected filename.
// Fl_File_Chooser::up() - Go up one directory.
// Fl_File_Chooser::newdir() - Make a new directory.
// Fl_File_Chooser::rescan() - Rescan the current directory.
// Fl_File_Chooser::fileListCB() - Handle clicks (and double-clicks) in the
// FileBrowser.
// Fl_File_Chooser::fileNameCB() - Handle text entry in the FileBrowser.
//
//
// Include necessary headers.
//
#include <FL/Fl_File_Chooser.H>
#include <FL/filename.H>
#include <FL/fl_ask.H>
#include <FL/x.H>
#include <stdio.h>
#include <stdlib.h>
#include "flstring.h"
#include <ctype.h>
#include <errno.h>
#include <sys/types.h>
#include <sys/stat.h>
#if defined(WIN32) && ! defined (__CYGWIN__)
# include <direct.h>
# include <io.h>
#else
# include <unistd.h>
# include <pwd.h>
#endif /* WIN32 */
//
// File chooser label strings...
//
const char *Fl_File_Chooser::directory_label = "Directory:";
const char *Fl_File_Chooser::filename_label = "Filename:";
const char *Fl_File_Chooser::filter_label = "New Filter?";
Fl_File_Sort_F *Fl_File_Chooser::sort = fl_numericsort;
//
// 'Fl_File_Chooser::directory()' - Set the directory in the file chooser.
//
void
Fl_File_Chooser::directory(const char *d)// I - Directory to change to
{
char pathname[1024], // Full path of directory
*pathptr, // Pointer into full path
*dirptr; // Pointer into directory
int levels; // Number of levels in directory
// printf("Fl_File_Chooser::directory(\"%s\")\n", d == NULL ? "(null)" : d);
// NULL == current directory
if (d == NULL)
d = ".";
if (d[0] != '\0')
{
// Make the directory absolute...
#if (defined(WIN32) && ! defined(__CYGWIN__))|| defined(__EMX__)
if (d[0] != '/' && d[0] != '\\' && d[1] != ':')
#else
if (d[0] != '/' && d[0] != '\\')
#endif /* WIN32 || __EMX__ */
fl_filename_absolute(directory_, d);
else
strlcpy(directory_, d, sizeof(directory_));
// Strip any trailing slash and/or period...
dirptr = directory_ + strlen(directory_) - 1;
if (*dirptr == '.')
*dirptr-- = '\0';
if ((*dirptr == '/' || *dirptr == '\\') && dirptr > directory_)
*dirptr = '\0';
}
else
directory_[0] = '\0';
// Clear the directory menu and fill it as needed...
dirMenu->clear();
#if (defined(WIN32) && ! defined(__CYGWIN__)) || defined(__EMX__)
dirMenu->add("My Computer");
#else
dirMenu->add("File Systems");
#endif /* WIN32 || __EMX__ */
levels = 0;
for (dirptr = directory_, pathptr = pathname; *dirptr != '\0';)
{
if (*dirptr == '/' || *dirptr == '\\')
{
// Need to quote the slash first, and then add it to the menu...
*pathptr++ = '\\';
*pathptr++ = '/';
*pathptr++ = '\0';
dirptr ++;
dirMenu->add(pathname);
levels ++;
pathptr = pathname;
}
else
*pathptr++ = *dirptr++;
}
if (pathptr > pathname)
{
*pathptr = '\0';
dirMenu->add(pathname);
levels ++;
}
dirMenu->value(levels);
// Rescan the directory...
rescan();
}
//
// 'Fl_File_Chooser::count()' - Return the number of selected files.
//
int // O - Number of selected files
Fl_File_Chooser::count()
{
int i; // Looping var
int count; // Number of selected files
const char *filename; // Filename in input field or list
char pathname[1024]; // Full path to file
if (!(type_ & MULTI))
{
// Check to see if the file name input field is blank...
filename = fileName->value();
if (filename == NULL || filename[0] == '\0')
return (0);
// Is the file name a directory?
if (directory_[0] != '\0')
snprintf(pathname, sizeof(pathname), "%s/%s", directory_, filename);
else
strlcpy(pathname, filename, sizeof(pathname));
if (fl_filename_isdir(pathname))
return (0);
else
return (1);
}
for (i = 1, count = 0; i <= fileList->size(); i ++)
if (fileList->selected(i))
{
// See if this file is a directory...
filename = (char *)fileList->text(i);
if (directory_[0] != '\0')
snprintf(pathname, sizeof(pathname), "%s/%s", directory_, filename);
else
strlcpy(pathname, filename, sizeof(pathname));
if (!fl_filename_isdir(pathname))
count ++;
}
return (count);
}
//
// 'Fl_File_Chooser::value()' - Return a selected filename.
//
const char * // O - Filename or NULL
Fl_File_Chooser::value(int f) // I - File number
{
int i; // Looping var
int count; // Number of selected files
const char *name; // Current filename
static char pathname[1024]; // Filename + directory
if (!(type_ & MULTI))
{
name = fileName->value();
if (name[0] == '\0') return NULL;
else if (fl_filename_isdir(name)) {
if (type_ & DIRECTORY) return name;
else return NULL;
} else return name;
}
for (i = 1, count = 0; i <= fileList->size(); i ++)
if (fileList->selected(i))
{
// See if this file is a directory...
name = fileList->text(i);
if (directory_[0]) {
snprintf(pathname, sizeof(pathname), "%s/%s", directory_, name);
} else {
strlcpy(pathname, name, sizeof(pathname));
}
if (!fl_filename_isdir(pathname))
{
// Nope, see if this this is "the one"...
count ++;
if (count == f)
return ((const char *)pathname);
}
}
return (NULL);
}
//
// 'Fl_File_Chooser::value()' - Set the current filename.
//
void
Fl_File_Chooser::value(const char *filename) // I - Filename + directory
{
int i, // Looping var
count; // Number of items in list
char *slash; // Directory separator
char pathname[1024]; // Local copy of filename
// printf("Fl_File_Chooser::value(\"%s\")\n", filename == NULL ? "(null)" : filename);
// See if the filename is the "My System" directory...
if (filename == NULL || !filename[0]) {
// Yes, just change the current directory...
directory(filename);
fileName->value("");
okButton->deactivate();
return;
}
// Switch to single-selection mode as needed
if (type_ & MULTI)
type(SINGLE);
// See if there is a directory in there...
fl_filename_absolute(pathname, sizeof(pathname), filename);
if ((slash = strrchr(pathname, '/')) == NULL)
slash = strrchr(pathname, '\\');
if (slash != NULL)
{
// Yes, change the display to the directory...
*slash++ = '\0';
directory(pathname);
}
else
{
directory(".");
slash = pathname;
}
// Set the input field to the absolute path...
if (slash > pathname) slash[-1] = '/';
fileName->value(pathname);
fileName->position(0, strlen(pathname));
okButton->activate();
// Then find the file in the file list and select it...
count = fileList->size();
fileList->deselect(0);
fileList->redraw();
for (i = 1; i <= count; i ++)
#if defined(WIN32) || defined(__EMX__)
if (strcasecmp(fileList->text(i), slash) == 0) {
#else
if (strcmp(fileList->text(i), slash) == 0) {
#endif // WIN32 || __EMX__
// printf("Selecting line %d...\n", i);
fileList->topline(i);
fileList->select(i);
break;
}
}
//
// 'Fl_File_Chooser::up()' - Go up one directory.
//
void
Fl_File_Chooser::up()
{
char *slash; // Trailing slash
if ((slash = strrchr(directory_, '/')) == NULL)
slash = strrchr(directory_, '\\');
if (directory_[0] != '\0')
dirMenu->value(dirMenu->value() - 1);
if (slash != NULL)
*slash = '\0';
else
{
upButton->deactivate();
directory_[0] = '\0';
}
rescan();
}
//
// 'Fl_File_Chooser::newdir()' - Make a new directory.
//
void
Fl_File_Chooser::newdir()
{
const char *dir; // New directory name
char pathname[1024]; // Full path of directory
// Get a directory name from the user
if ((dir = fl_input("New Directory?", NULL)) == NULL)
return;
// Make it relative to the current directory as needed...
#if (defined(WIN32) && ! defined (__CYGWIN__)) || defined(__EMX__)
if (dir[0] != '/' && dir[0] != '\\' && dir[1] != ':')
#else
if (dir[0] != '/' && dir[0] != '\\')
#endif /* WIN32 || __EMX__ */
snprintf(pathname, sizeof(pathname), "%s/%s", directory_, dir);
else
strlcpy(pathname, dir, sizeof(pathname));
// Create the directory; ignore EEXIST errors...
#if defined(WIN32) && ! defined (__CYGWIN__)
if (mkdir(pathname))
#else
if (mkdir(pathname, 0777))
#endif /* WIN32 */
if (errno != EEXIST)
{
fl_alert("Unable to create directory!");
return;
}
// Show the new directory...
directory(pathname);
}
//
// 'Fl_File_Chooser::rescan()' - Rescan the current directory.
//
void
Fl_File_Chooser::rescan()
{
char pathname[1024]; // New pathname for filename field
// printf("Fl_File_Chooser::rescan(); directory = \"%s\"\n", directory_);
// Clear the current filename
strlcpy(pathname, directory_, sizeof(pathname));
if (pathname[strlen(pathname) - 1] != '/') {
strlcat(pathname, "/", sizeof(pathname));
}
fileName->value(pathname);
okButton->deactivate();
// Build the file list...
fileList->load(directory_, sort);
}
//
// 'Fl_File_Chooser::fileListCB()' - Handle clicks (and double-clicks) in the
// FileBrowser.
//
void
Fl_File_Chooser::fileListCB()
{
char *filename, // New filename
pathname[1024]; // Full pathname to file
filename = (char *)fileList->text(fileList->value());
if (!filename)
return;
if (directory_[0] != '\0')
snprintf(pathname, sizeof(pathname), "%s/%s", directory_, filename);
else
strlcpy(pathname, filename, sizeof(pathname));
if (Fl::event_clicks())
{
#if (defined(WIN32) && ! defined(__CYGWIN__)) || defined(__EMX__)
if ((strlen(pathname) == 2 && pathname[1] == ':') ||
fl_filename_isdir(pathname))
#else
if (fl_filename_isdir(pathname))
#endif /* WIN32 || __EMX__ */
{
directory(pathname);
upButton->activate();
}
else
{
// Do any callback that is registered...
if (callback_)
(*callback_)(this, data_);
// Hide the window...
window->hide();
}
}
else
{
// Strip any trailing slash from the directory name...
filename = pathname + strlen(pathname) - 1;
if (*filename == '/') *filename = '\0';
fileName->value(pathname);
if (!fl_filename_isdir(pathname) || (type_ & DIRECTORY))
okButton->activate();
}
}
//
// 'Fl_File_Chooser::fileNameCB()' - Handle text entry in the FileBrowser.
//
void
Fl_File_Chooser::fileNameCB()
{
char *filename, // New filename
*slash, // Pointer to trailing slash
pathname[1024], // Full pathname to file
matchname[256]; // Matching filename
int i, // Looping var
min_match, // Minimum number of matching chars
max_match, // Maximum number of matching chars
num_files, // Number of files in directory
first_line; // First matching line
const char *file; // File from directory
// Get the filename from the text field...
filename = (char *)fileName->value();
if (filename == NULL || filename[0] == '\0')
{
okButton->deactivate();
return;
}
// Expand ~ and $ variables as needed...
if (strchr(filename, '~') || strchr(filename, '$')) {
fl_filename_expand(pathname, sizeof(pathname), filename);
filename = pathname;
value(pathname);
}
// Make sure we have an absolute path...
#if (defined(WIN32) && !defined(__CYGWIN__)) || defined(__EMX__)
if (directory_[0] != '\0' && filename[0] != '/' &&
filename[0] != '\\' &&
!(isalpha(filename[0]) && filename[1] == ':')) {
#else
if (directory_[0] != '\0' && filename[0] != '/') {
#endif /* WIN32 || __EMX__ */
fl_filename_absolute(pathname, sizeof(pathname), filename);
value(pathname);
} else if (filename != pathname) {
// Finally, make sure that we have a writable copy...
strlcpy(pathname, filename, sizeof(pathname));
}
filename = pathname;
// Now process things according to the key pressed...
if (Fl::event_key() == FL_Enter)
{
// Enter pressed - select or change directory...
#if (defined(WIN32) && ! defined(__CYGWIN__)) || defined(__EMX__)
if ((strlen(pathname) == 2 && pathname[1] == ':') ||
fl_filename_isdir(pathname))
#else
if (fl_filename_isdir(pathname))
#endif /* WIN32 || __EMX__ */
directory(pathname);
else if ((type_ & CREATE) || access(pathname, 0) == 0)
{
// New file or file exists... If we are in multiple selection mode,
// switch to single selection mode...
if (type_ & MULTI)
type(SINGLE);
// Do any callback that is registered...
if (callback_)
(*callback_)(this, data_);
// Hide the window to signal things are done...
window->hide();
}
else
{
// File doesn't exist, so beep at and alert the user...
fl_alert("Please choose an existing file!");
}
}
else if (Fl::event_key() != FL_Delete &&
Fl::event_key() != FL_BackSpace)
{
// Check to see if the user has entered a directory...
if ((slash = strrchr(pathname, '/')) == NULL)
slash = strrchr(pathname, '\\');
if (slash != NULL)
{
// Yes, change directories if necessary...
if (slash > pathname) // Special case for "/"
*slash++ = '\0';
else
slash++;
filename = slash;
#if defined(WIN32) || defined(__EMX__)
if (strcasecmp(pathname, directory_)) {
#else
if (strcmp(pathname, directory_)) {
#endif // WIN32 || __EMX__
int p = fileName->position();
int m = fileName->mark();
directory(pathname);
fileName->position(p, m);
}
}
// Other key pressed - do filename completion as possible...
num_files = fileList->size();
min_match = strlen(filename);
max_match = 100000;
first_line = 0;
for (i = 1; i <= num_files && max_match > min_match; i ++)
{
file = fileList->text(i);
#if (defined(WIN32) && ! defined(__CYGWIN__)) || defined(__EMX__)
if (strnicmp(filename, file, min_match) == 0)
#else
if (strncmp(filename, file, min_match) == 0)
#endif // WIN32 || __EMX__
{
// OK, this one matches; check against the previous match
if (max_match == 100000)
{
// First match; copy stuff over...
strlcpy(matchname, file, sizeof(matchname));
max_match = strlen(matchname);
// Strip trailing /, if any...
if (matchname[max_match - 1] == '/')
{
max_match --;
matchname[max_match] = '\0';
}
// And then make sure that the item is visible
fileList->topline(i);
first_line = i;
}
else
{
// Succeeding match; compare to find maximum string match...
while (max_match > min_match)
#if (defined(WIN32) && ! defined(__CYGWIN__)) || defined(__EMX__)
if (strnicmp(file, matchname, max_match) == 0)
#else
if (strncmp(file, matchname, max_match) == 0)
#endif // WIN32 || __EMX__
break;
else
max_match --;
// Truncate the string as needed...
matchname[max_match] = '\0';
}
}
}
// If we have any matches, add them to the input field...
if (first_line > 0 && min_match == max_match &&
max_match == (int)strlen(fileList->text(first_line))) {
// This is the only possible match...
fileList->deselect(0);
fileList->select(first_line);
fileList->redraw();
}
else if (max_match > min_match && max_match != 100000)
{
// Add the matching portion...
fileName->replace(filename - pathname, filename - pathname + min_match,
matchname);
// Highlight it with the cursor at the end of the selection so
// s/he can press the right arrow to accept the selection
// (Tab and End also do this for both cases.)
fileName->position(filename - pathname + max_match,
filename - pathname + min_match);
}
else if (max_match == 0) {
fileList->deselect(0);
fileList->redraw();
}
// See if we need to enable the OK button...
if ((type_ & CREATE || access(fileName->value(), 0) == 0) &&
(!fl_filename_isdir(fileName->value()) || type_ & DIRECTORY))
okButton->activate();
else
okButton->deactivate();
} else {
// FL_Delete or FL_BackSpace
fileList->deselect(0);
fileList->redraw();
okButton->deactivate();
}
}
//
// End of "$Id: Fl_File_Chooser2.cxx,v 1.1.2.13 2002/05/16 12:47:43 easysw Exp $".
//