claws-mail/src/exporthtml.c
2021-09-15 16:54:34 -07:00

1179 lines
31 KiB
C

/*
* Sylpheed -- a GTK+ based, lightweight, and fast e-mail client
* Copyright (C) 2002-2012 Match Grun and the Claws Mail team
*
* 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 3 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, see <http://www.gnu.org/licenses/>.
*
*/
/*
* Export address book to HTML file.
*/
#ifdef HAVE_CONFIG_H
# include "config.h"
#include "claws-features.h"
#endif
#ifdef USE_PTHREAD
#include <pthread.h>
#endif
#include <sys/stat.h>
#include <dirent.h>
#include <errno.h>
#include <time.h>
#include <string.h>
#include <glib.h>
#include <glib/gi18n.h>
#include "mgutils.h"
#include "utils.h"
#include "exporthtml.h"
#include "xmlprops.h"
#include "file-utils.h"
#ifdef MKDIR_TAKES_ONE_ARG
#undef mkdir
#define mkdir(a,b) mkdir(a)
#endif
#define DFL_DIR_CLAWS_OUT "claws-mail-out"
#define DFL_FILE_CLAWS_OUT "addressbook.html"
#define FMT_BUFSIZE 2048
#define SC_HTML_SPACE "&nbsp;"
#define BORDER_SIZE 2
#define CELL_PADDING 2
#define CELL_SPACING 2
#define CHAR_ENCODING "UTF-8"
/* Stylesheet names */
#define FILENAME_NONE ""
#define FILENAME_DEFAULT "claws-mail.css"
#define FILENAME_FULL "full.css"
#define FILENAME_CUSTOM "custom.css"
#define FILENAME_CUSTOM2 "custom2.css"
#define FILENAME_CUSTOM3 "custom3.css"
#define FILENAME_CUSTOM4 "custom4.css"
/* Settings - properties */
#define EXML_PROPFILE_NAME "exporthtml.xml"
#define EXMLPROP_DIRECTORY "directory"
#define EXMLPROP_FILE "file"
#define EXMLPROP_STYLESHEET "stylesheet"
#define EXMLPROP_FMT_NAME "format-full-name"
#define EXMLPROP_FMT_EMAIL "format-email-links"
#define EXMLPROP_FMT_ATTRIB "format-attributes"
#define EXMLPROP_BANDING "color-banding"
#define EXMLPROP_VALUE_YES "y"
#define EXMLPROP_VALUE_NO "n"
static gchar *_idTagRowEven_ = "tab-row0";
static gchar *_idTagRowOdd_ = "tab-row1";
/*
* Header entry.
*/
typedef struct _StylesheetEntry StylesheetEntry;
struct _StylesheetEntry {
gchar *fileName;
gint id;
gboolean dflValue;
};
/*
* Build stylesheet entry.
* Enter: ctl Export control data.
* file Filename.
* id File id.
* dfl Default flag.
*/
static void exporthtml_build_entry(
ExportHtmlCtl *ctl, const gchar *file, const gint id,
const gboolean dfl )
{
StylesheetEntry *entry;
entry = g_new0( StylesheetEntry, 1 );
entry->fileName = g_strdup( file );
entry->id = id;
entry->dflValue = dfl;
ctl->listStyle = g_list_append( ctl->listStyle, entry );
}
/*
* Free up object by releasing internal memory.
* Enter: ctl Export control data.
*/
ExportHtmlCtl *exporthtml_create( void ) {
ExportHtmlCtl *ctl = g_new0( ExportHtmlCtl, 1 );
ctl->path = NULL;
ctl->dirOutput = NULL;
ctl->fileHtml = NULL;
ctl->encoding = g_strconcat(CHAR_ENCODING, NULL);
ctl->stylesheet = EXPORT_HTML_ID_NONE;
ctl->nameFormat = EXPORT_HTML_FIRST_LAST;
ctl->banding = FALSE;
ctl->linkEMail = FALSE;
ctl->showAttribs = FALSE;
ctl->retVal = MGU_SUCCESS;
ctl->listStyle = NULL;
ctl->rcCreate = 0;
ctl->settingsFile = g_strconcat(
get_rc_dir(), G_DIR_SEPARATOR_S, EXML_PROPFILE_NAME, NULL );
/* Build stylesheet list */
exporthtml_build_entry(
ctl, FILENAME_NONE, EXPORT_HTML_ID_NONE, FALSE );
exporthtml_build_entry(
ctl, FILENAME_DEFAULT, EXPORT_HTML_ID_DEFAULT, TRUE );
exporthtml_build_entry(
ctl, FILENAME_FULL, EXPORT_HTML_ID_FULL, FALSE );
exporthtml_build_entry(
ctl, FILENAME_CUSTOM, EXPORT_HTML_ID_CUSTOM, FALSE );
exporthtml_build_entry(
ctl, FILENAME_CUSTOM2, EXPORT_HTML_ID_CUSTOM2, FALSE );
exporthtml_build_entry(
ctl, FILENAME_CUSTOM3, EXPORT_HTML_ID_CUSTOM3, FALSE );
exporthtml_build_entry(
ctl, FILENAME_CUSTOM4, EXPORT_HTML_ID_CUSTOM4, FALSE );
return ctl;
}
/*
* Free up object by releasing internal memory.
* Enter: ctl Export control data.
*/
void exporthtml_free( ExportHtmlCtl *ctl ) {
GList *node;
StylesheetEntry *entry;
cm_return_if_fail( ctl != NULL );
/* Free stylesheet list */
node = ctl->listStyle;
while( node ) {
entry = ( StylesheetEntry * ) node->data;
g_free( entry->fileName );
entry->fileName = NULL;
entry->id = 0;
entry->dflValue = FALSE;
g_free( entry );
node->data = NULL;
node = g_list_next( node );
}
g_list_free( ctl->listStyle );
ctl->listStyle = NULL;
g_free( ctl->path );
g_free( ctl->fileHtml );
g_free( ctl->encoding );
g_free( ctl->dirOutput );
g_free( ctl->settingsFile );
/* Clear pointers */
ctl->path = NULL;
ctl->dirOutput = NULL;
ctl->fileHtml = NULL;
ctl->encoding = NULL;
ctl->stylesheet = EXPORT_HTML_ID_NONE;
ctl->nameFormat = EXPORT_HTML_FIRST_LAST;
ctl->banding = FALSE;
ctl->linkEMail = FALSE;
ctl->showAttribs = FALSE;
ctl->retVal = MGU_SUCCESS;
ctl->rcCreate = 0;
/* Now release object */
g_free( ctl );
}
/*
* Find style entry.
* Enter: ctl Export control data.
* Return: Stylesheet object, or NULL if nothing found. If a default entry is
* found in list, it will be returned.
*/
static StylesheetEntry *exporthtml_find_stylesheet( ExportHtmlCtl *ctl ) {
StylesheetEntry *retVal = NULL;
StylesheetEntry *entry;
GList *node;
node = ctl->listStyle;
while( node ) {
entry = ( StylesheetEntry * ) node->data;
if( entry->id == ctl->stylesheet ) return entry;
if( entry->dflValue ) retVal = entry;
node = g_list_next( node );
}
return retVal;
}
void exporthtml_set_stylesheet( ExportHtmlCtl *ctl, const gint value ) {
cm_return_if_fail( ctl != NULL );
ctl->stylesheet = value;
}
void exporthtml_set_name_format( ExportHtmlCtl *ctl, const gint value ) {
cm_return_if_fail( ctl != NULL );
ctl->nameFormat = value;
}
void exporthtml_set_banding( ExportHtmlCtl *ctl, const gboolean value ) {
cm_return_if_fail( ctl != NULL );
ctl->banding = value;
}
void exporthtml_set_link_email( ExportHtmlCtl *ctl, const gboolean value ) {
cm_return_if_fail( ctl != NULL );
ctl->linkEMail = value;
}
void exporthtml_set_attributes( ExportHtmlCtl *ctl, const gboolean value ) {
cm_return_if_fail( ctl != NULL );
ctl->showAttribs = value;
}
/*
* Create default CSS file.
* Enter: fileSpec File to create.
* Return: Status code.
*/
static gint exporthtml_create_css_dfl( const gchar *fileSpec ) {
FILE *cssFile;
cssFile = claws_fopen( fileSpec, "rb" );
if( cssFile ) {
claws_fclose( cssFile );
return MGU_SUCCESS;
}
cssFile = claws_fopen( fileSpec, "wb" );
if( ! cssFile ) {
return MGU_OPEN_FILE;
}
fprintf( cssFile, "body {\n\tbackground: #ffffe0;\n" );
fprintf( cssFile, "\tfont-family: lucida, helvetica, sans-serif;\n" );
fprintf( cssFile, "\tfont-size: 10pt;\n" );
fprintf( cssFile, "}\n" );
fprintf( cssFile, "h1 {\n" );
fprintf( cssFile, "\tcolor: #000000;\n" );
fprintf( cssFile, "\ttext-align: center;\n" );
fprintf( cssFile, "}\n" );
fprintf( cssFile, "th {\n" );
fprintf( cssFile, "\tfont-size: 10pt;\n" );
fprintf( cssFile, "}\n" );
fprintf( cssFile, "td {\n" );
fprintf( cssFile, "\tfont-size: 10pt;\n" );
fprintf( cssFile, "}\n" );
fprintf( cssFile, ".fmt-folder {\n" );
fprintf( cssFile, "\tcolor: #0000ff;\n" );
fprintf( cssFile, "\tfont-size: 18pt;\n" );
fprintf( cssFile, "\tfont-weight: bold;\n" );
fprintf( cssFile, "}\n" );
fprintf( cssFile, ".tab-head {\n" );
fprintf( cssFile, "\tbackground: #80c0f0;\n" );
fprintf( cssFile, "}\n" );
fprintf( cssFile, ".tab-dn {\n" );
fprintf( cssFile, "}\n" );
fprintf( cssFile, ".tab-addr {\n" );
fprintf( cssFile, "\tfont-style: italic;\n" );
fprintf( cssFile, "}\n" );
fprintf( cssFile, ".tab-email {\n" );
fprintf( cssFile, "\tfont-weight: bold;\n" );
fprintf( cssFile, "\tfont-style: italic;\n" );
fprintf( cssFile, "}\n" );
fprintf( cssFile, ".tab-fn {\n" );
fprintf( cssFile, "}\n" );
fprintf( cssFile, ".tab-attr {\n" );
fprintf( cssFile, "}\n" );
claws_safe_fclose( cssFile );
return MGU_SUCCESS;
}
/*
* Create full CSS file.
* Enter: fileSpec File to create.
* Return: Status code.
*/
static gint exporthtml_create_css_full( const gchar *fileSpec ) {
FILE *cssFile;
cssFile = claws_fopen( fileSpec, "rb" );
if( cssFile ) {
claws_fclose( cssFile );
return MGU_SUCCESS;
}
cssFile = claws_fopen( fileSpec, "wb" );
if( ! cssFile ) {
return MGU_OPEN_FILE;
}
fprintf( cssFile, "body {\n\tbackground: #ffffe0;\n" );
fprintf( cssFile, "\tfont-family: lucida, helvetica, sans-serif;\n" );
fprintf( cssFile, "\tfont-size: 10pt;\n" );
fprintf( cssFile, "}\n" );
fprintf( cssFile, "h1 {\n" );
fprintf( cssFile, "\tcolor: #000000;\n" );
fprintf( cssFile, "\ttext-align: center;\n" );
fprintf( cssFile, "}\n" );
fprintf( cssFile, "th {\n" );
fprintf( cssFile, "\tfont-size: 10pt;\n" );
fprintf( cssFile, "}\n" );
fprintf( cssFile, "td {\n" );
fprintf( cssFile, "\tfont-size: 10pt;\n" );
fprintf( cssFile, "}\n" );
fprintf( cssFile, ".fmt-folder {\n" );
fprintf( cssFile, "\tcolor: #0000ff;\n" );
fprintf( cssFile, "\tfont-size: 18pt;\n" );
fprintf( cssFile, "\tfont-weight: bold;\n" );
fprintf( cssFile, "}\n" );
fprintf( cssFile, ".tab-head {\n" );
fprintf( cssFile, "\tbackground: #80c0f0;\n" );
fprintf( cssFile, "}\n" );
fprintf( cssFile, ".tab-row0 {\n" );
fprintf( cssFile, "\tbackground: #f0f0f0;\n" );
fprintf( cssFile, "}\n" );
fprintf( cssFile, ".tab-row1 {\n" );
fprintf( cssFile, "\tbackground: #d0d0d0;\n" );
fprintf( cssFile, "}\n" );
fprintf( cssFile, ".tab-dn {\n" );
fprintf( cssFile, "}\n" );
fprintf( cssFile, ".tab-addr {\n" );
fprintf( cssFile, "\tfont-style: italic;\n" );
fprintf( cssFile, "}\n" );
fprintf( cssFile, ".tab-email {\n" );
fprintf( cssFile, "\tfont-weight: bold;\n" );
fprintf( cssFile, "\tfont-style: italic;\n" );
fprintf( cssFile, "}\n" );
fprintf( cssFile, ".tab-fn {\n" );
fprintf( cssFile, "}\n" );
fprintf( cssFile, ".tab-attr {\n" );
fprintf( cssFile, "}\n" );
claws_safe_fclose( cssFile );
return MGU_SUCCESS;
}
/*
* Create stylesheet files.
* Enter: ctl Export control data.
*/
static void exporthtml_create_css_files( ExportHtmlCtl *ctl ) {
gchar *fileSpec;
GList *node;
node = ctl->listStyle;
while( node ) {
StylesheetEntry *entry = node->data;
node = g_list_next( node );
if( strlen( entry->fileName ) ) {
fileSpec = g_strconcat(
ctl->dirOutput, G_DIR_SEPARATOR_S,
entry->fileName, NULL );
if( entry->id == EXPORT_HTML_ID_DEFAULT ) {
exporthtml_create_css_dfl( fileSpec );
}
else if( entry->id != EXPORT_HTML_ID_NONE ) {
exporthtml_create_css_full( fileSpec );
}
g_free( fileSpec );
}
}
}
/*
* Comparison using linked list elements.
*/
static gint exporthtml_compare_name(
gconstpointer ptr1, gconstpointer ptr2 )
{
const AddrItemObject *item1 = ptr1;
const AddrItemObject *item2 = ptr2;
const gchar *name1 = NULL, *name2 = NULL;
if( item1 ) name1 = ADDRITEM_NAME( item1 );
if( item2 ) name2 = ADDRITEM_NAME( item2 );
if( ! name1 ) return ( name2 != NULL );
if( ! name2 ) return -1;
return g_utf8_collate( name1, name2 );
}
/*
* Comparison using linked list elements.
*/
static gint exporthtml_compare_email(
gconstpointer ptr1, gconstpointer ptr2 )
{
const ItemEMail *email1 = ptr1;
const ItemEMail *email2 = ptr2;
const gchar *name1 = NULL, *name2 = NULL;
if( email1 ) name1 = email1->address;
if( email2 ) name2 = email2->address;
if( ! name1 ) return ( name2 != NULL );
if( ! name2 ) return -1;
return g_utf8_collate( name1, name2 );
}
/*
* Comparison using linked list elements.
*/
static gint exporthtml_compare_attrib(
gconstpointer ptr1, gconstpointer ptr2 )
{
const UserAttribute *attr1 = ptr1;
const UserAttribute *attr2 = ptr2;
const gchar *name1 = NULL, *name2 = NULL;
if( attr1 ) name1 = attr1->name;
if( attr2 ) name2 = attr2->name;
if( ! name1 ) return ( name2 != NULL );
if( ! name2 ) return -1;
return g_utf8_collate( name1, name2 );
}
/*
* Build sorted list of named items.
* Enter: list List of items to sorted.
* Return: Sorted list.
* Note: List should freed after use. Items referenced by list should not be
* freed since they are managed by the address cache.
*/
static GList *exporthtml_sort_name( const GList *list ) {
const GList *node;
GList *sorted = NULL;
node = list;
while( node ) {
sorted = g_list_insert_sorted(
sorted, node->data, exporthtml_compare_name );
node = g_list_next( node );
}
return sorted;
}
/*
* Build sorted list of email items.
* Enter: list List of E-Mail items to sorted.
* Return: Sorted list.
* Note: List should freed after use. Items referenced by list should not be
* freed since they are managed by the address cache.
*/
static GList *exporthtml_sort_email( const GList *list ) {
const GList *node;
GList *sorted = NULL;
node = list;
while( node ) {
sorted = g_list_insert_sorted(
sorted, node->data, exporthtml_compare_email );
node = g_list_next( node );
}
return sorted;
}
/*
* Build sorted list of attributes.
* Enter: list List of items to sorted.
* Return: Sorted list.
* Note: List should freed after use. Items referenced by list should not be
* freed since they are managed by the address cache.
*/
static GList *exporthtml_sort_attrib( const GList *list ) {
const GList *node;
GList *sorted = NULL;
sorted = NULL;
node = list;
while( node ) {
sorted = g_list_insert_sorted(
sorted, node->data, exporthtml_compare_attrib );
node = g_list_next( node );
}
return sorted;
}
/*
* Format a list of E-Mail addresses.
* Enter: ctl Export control data.
* stream Output stream.
* listEMail List of addresses.
* sortFlag Set to TRUE if address list should be sorted.
*/
static void exporthtml_fmt_email(
ExportHtmlCtl *ctl, FILE *stream, const GList *listEMail,
gboolean sortFlag )
{
const GList *node;
GList *list;
gchar *name;
if( listEMail == NULL ) {
fprintf( stream, SC_HTML_SPACE );
return;
}
list = NULL;
if( sortFlag ) {
node = list = exporthtml_sort_email( listEMail );
}
else {
node = listEMail;
}
while( node ) {
ItemEMail *email = ( ItemEMail * ) node->data;
node = g_list_next( node );
name = ADDRITEM_NAME( email );
if( name ) {
fprintf( stream, "%s ", name );
}
if( ctl->linkEMail ) {
fprintf( stream, "<a href=\"mailto:%s\">",
email->address );
}
fprintf( stream, "<span class=\"tab-email\">" );
fprintf( stream, "%s", email->address );
fprintf( stream, "</span>" );
if( ctl->linkEMail ) {
fprintf( stream, "</a>" );
}
if( email->remarks ) {
if( strlen( email->remarks ) ) {
fprintf( stream, " (%s)", email->remarks );
}
}
fprintf( stream, "<br>\n" );
}
g_list_free( list );
}
/*
* Format groups in an address book folder.
* Enter: ctl Export control data.
* stream Output stream.
* folder Folder.
* prevFlag If FALSE, list of persons were output.
* Return: TRUE if no groups were formatted.
*/
static gboolean exporthtml_fmt_group(
ExportHtmlCtl *ctl, FILE *stream, const ItemFolder *folder,
gboolean prevFlag )
{
gboolean retVal, band;
GList *node, *list;
const gchar *tagName;
retVal = TRUE;
if( folder->listGroup == NULL ) return retVal;
/* Write separator */
if( ! prevFlag ) {
fprintf( stream, "<br>\n" );
}
/* Write table headers */
fprintf( stream, "<table" );
fprintf( stream, " border=\"%d\"", BORDER_SIZE );
fprintf( stream, " cellpadding=\"%d\"", CELL_PADDING );
fprintf( stream, " cellspacing=\"%d\"", CELL_SPACING );
fprintf( stream, ">\n" );
fprintf( stream, "<tr class=\"tab-head\">\n" );
fprintf( stream, " <th width=\"200\">" );
fprintf( stream, "%s", _( "Group Name" ) );
fprintf( stream, "</th>\n" );
fprintf( stream, " <th width=\"300\">" );
fprintf( stream, "%s", _( "Email Address" ) );
fprintf( stream, "</th>\n" );
fprintf( stream, "</tr>\n" );
list = exporthtml_sort_name( folder->listGroup );
band = FALSE;
node = list;
while( node ) {
AddrItemObject *aio = node->data;
if( aio && aio->type == ITEMTYPE_GROUP ) {
ItemGroup *group = ( ItemGroup * ) aio;
fprintf( stream, "<tr valign=\"top\"" );
if( ctl->banding ) {
if( band ) {
tagName = _idTagRowOdd_;
}
else {
tagName = _idTagRowEven_;
}
fprintf( stream, " class=\"%s\"", tagName );
band = ! band;
}
fprintf( stream, "\">\n" );
fprintf( stream, " <td class=\"tab-dn\">" );
fprintf( stream, "%s", ADDRITEM_NAME( group ) );
fprintf( stream, "</td>\n" );
fprintf( stream, " <td class=\"tab-addr\">" );
exporthtml_fmt_email( ctl, stream, group->listEMail, TRUE );
fprintf( stream, "</td>\n" );
fprintf( stream, "</tr>\n" );
retVal = FALSE;
}
node = g_list_next( node );
}
g_list_free( list );
fprintf( stream, "</table>\n" );
return retVal;
}
/*
* Format a list of E-Mail addresses.
* Enter: ctl Export control data.
* stream Output stream.
* listAttr List of attributes.
*/
static void exporthtml_fmt_attribs(
ExportHtmlCtl *ctl, FILE *stream, const GList *listAttr )
{
const GList *node;
GList *list;
if( listAttr == NULL ) {
fprintf( stream, SC_HTML_SPACE );
return;
}
fprintf( stream, "<table border=\"0\">\n" );
node = list = exporthtml_sort_attrib( listAttr );
while( node ) {
UserAttribute *attr = ( UserAttribute * ) node->data;
node = g_list_next( node );
fprintf( stream, "<tr valign=\"top\">" );
fprintf( stream, "<td align=\"right\">%s:</td>", attr->name );
fprintf( stream, "<td>%s</td>", attr->value );
fprintf( stream, "</tr>\n" );
}
g_list_free( list );
fprintf( stream, "</table>" );
}
/*
* Format full name.
* Enter: ctl Export control data.
* buf Output buffer.
* person Person to format.
*/
static void exporthtml_fmt_fullname(
ExportHtmlCtl *ctl, gchar *buf, const ItemPerson *person )
{
gboolean flag;
if( ctl->nameFormat == EXPORT_HTML_LAST_FIRST ) {
flag = FALSE;
if( person->lastName ) {
if( *person->lastName ) {
strcat( buf, " " );
strcat( buf, person->lastName );
flag = TRUE;
}
}
if( person->firstName ) {
if( *person->firstName ) {
if( flag ) {
strcat( buf, ", " );
}
strcat( buf, person->firstName );
}
}
}
else {
if( person->firstName ) {
if( *person->firstName ) {
strcat( buf, person->firstName );
}
}
if( person->lastName ) {
if( *person->lastName ) {
strcat( buf, " " );
strcat( buf, person->lastName );
}
}
}
g_strstrip( buf );
flag = FALSE;
if( *buf ) flag = TRUE;
if( person->nickName ) {
if( strlen( person->nickName ) ) {
if( flag ) {
strcat( buf, " (" );
}
strcat( buf, person->nickName );
if( flag ) {
strcat( buf, ")" );
}
}
}
g_strstrip( buf );
}
/*
* Format persons in an address book folder.
* Enter: ctl Export control data.
* stream Output stream.
* folder Folder.
* Return: TRUE if no persons were formatted.
*/
static gboolean exporthtml_fmt_person(
ExportHtmlCtl *ctl, FILE *stream, const ItemFolder *folder )
{
gboolean retVal, band;
GList *node, *list;
gchar buf[ FMT_BUFSIZE ];
const gchar *tagName;
retVal = TRUE;
if( folder->listPerson == NULL ) return retVal;
/* Write table headers */
fprintf( stream, "<table" );
fprintf( stream, " border=\"%d\"", BORDER_SIZE );
fprintf( stream, " cellpadding=\"%d\"", CELL_PADDING );
fprintf( stream, " cellspacing=\"%d\"", CELL_SPACING );
fprintf( stream, ">\n" );
fprintf( stream, "<tr class=\"tab-head\">\n" );
fprintf( stream, " <th width=\"200\">" );
fprintf( stream, "%s", _( "Display Name" ) );
fprintf( stream, "</th>\n" );
fprintf( stream, " <th width=\"300\">" );
fprintf( stream, "%s", _( "Email Address" ) );
fprintf( stream, "</th>\n" );
fprintf( stream, " <th width=\"200\">" );
fprintf( stream, "%s", _( "Full Name" ) );
fprintf( stream, "</th>\n" );
if( ctl->showAttribs ) {
fprintf( stream, " <th width=\"250\">" );
fprintf( stream, "%s", _( "Attributes" ) );
fprintf( stream, "</th>\n" );
}
fprintf( stream, "</tr>\n" );
band = FALSE;
node = list = exporthtml_sort_name( folder->listPerson );
while( node ) {
AddrItemObject *aio = node->data;
if( aio && aio->type == ITEMTYPE_PERSON ) {
ItemPerson *person = ( ItemPerson * ) aio;
/* Format first/last/nick name */
*buf = '\0';
exporthtml_fmt_fullname( ctl, buf,person );
fprintf( stream, "<tr valign=\"top\"" );
if( ctl->banding ) {
if( band ) {
tagName = _idTagRowOdd_;
}
else {
tagName = _idTagRowEven_;
}
fprintf( stream, " class=\"%s\"", tagName );
band = ! band;
}
fprintf( stream, ">\n" );
fprintf( stream, " <td class=\"tab-dn\">" );
fprintf( stream, "%s", ADDRITEM_NAME( person ) );
fprintf( stream, "</td>\n" );
fprintf( stream, " <td class=\"tab-addr\">" );
exporthtml_fmt_email( ctl, stream, person->listEMail, FALSE );
fprintf( stream, "</td>\n" );
fprintf( stream, " <td class=\"tab-fn\">" );
if( *buf ) {
fprintf( stream, "%s", buf );
}
else {
fprintf( stream, "%s", SC_HTML_SPACE );
}
fprintf( stream, "</td>\n" );
if( ctl->showAttribs ) {
fprintf( stream, " <td class=\"tab-attr\">" );
exporthtml_fmt_attribs(
ctl, stream, person->listAttrib );
fprintf( stream, "</td>\n" );
}
fprintf( stream, "</tr>\n" );
retVal = FALSE;
}
node = g_list_next( node );
}
g_list_free( list );
fprintf( stream, "</table>\n" );
return retVal;
}
/*
* Format folder heirarchy.
* Enter: stream Output stream.
* list Heirarchy list.
*/
static void exporthtml_fmt_folderhead( FILE *stream, const GList *list ) {
const GList *node;
gboolean flag;
gchar *name;
flag = FALSE;
node = list;
while( node ) {
AddrItemObject *aio = node->data;
if( aio && aio->type == ITEMTYPE_FOLDER ) {
ItemFolder *folder = ( ItemFolder * ) aio;
name = ADDRITEM_NAME( folder );
if( name ) {
if( flag ) {
fprintf( stream, "&nbsp;&gt;&nbsp;" );
}
fprintf( stream, "%s", name );
flag = TRUE;
}
}
node = g_list_next( node );
}
}
/*
* Format an address book folder.
* Enter: ctl Export control data.
* stream Output stream.
* folder Folder.
*/
static void exporthtml_fmt_folder(
ExportHtmlCtl *ctl, FILE *stream, const ItemFolder *folder )
{
const GList *node;
GList *listHeir, *list;
const gchar *name;
gboolean ret1;
name = ADDRITEM_NAME( folder );
if( name ) {
listHeir = addritem_folder_path( folder, TRUE );
if( listHeir ) {
fprintf( stream, "<p class=\"fmt-folder\">" );
fprintf( stream, "%s: ", _( "Folder" ) );
exporthtml_fmt_folderhead( stream, listHeir );
fprintf( stream, "</p>\n" );
g_list_free( listHeir );
}
}
ret1 = exporthtml_fmt_person( ctl, stream, folder );
exporthtml_fmt_group( ctl, stream, folder, ret1 );
node = list = exporthtml_sort_name( folder->listFolder );
while( node ) {
AddrItemObject *aio = node->data;
if( aio && aio->type == ITEMTYPE_FOLDER ) {
ItemFolder *subFolder = ( ItemFolder * ) aio;
exporthtml_fmt_folder( ctl, stream, subFolder );
}
node = g_list_next( node );
}
if( list ) {
g_list_free( list );
}
}
/*
* Format header block.
* Enter: ctl Export control data.
* stream Output stream.
* title Page title.
*/
static void exporthtml_fmt_header(
ExportHtmlCtl *ctl, FILE *stream, gchar *title )
{
StylesheetEntry *entry;
entry = exporthtml_find_stylesheet( ctl );
fprintf( stream,
"<!DOCTYPE html PUBLIC \"-//W3C//DTD HTML 4.01 Transitional//EN\"\n" );
fprintf( stream,
"\"http://www.w3.org/TR/html4/loose.dtd\">\n" );
fprintf( stream, "<html>\n" );
fprintf( stream, "<head>\n" );
if( ctl->encoding && strlen( ctl->encoding ) > 0 ) {
fprintf( stream, "<meta " );
fprintf( stream, "http-equiv=\"Content-Type\" " );
fprintf( stream, "content=\"text/html; charset=%s\">\n",
ctl->encoding );
}
fprintf( stream, "<title>%s</title>\n", title );
if( entry != NULL ) {
if( entry->fileName && strlen( entry->fileName ) > 0 ) {
fprintf( stream, "<link " );
fprintf( stream, "rel=\"stylesheet\" " );
fprintf( stream, "type=\"text/css\" " );
fprintf( stream, "href=\"%s\" >\n", entry->fileName );
}
}
fprintf( stream, "</head>\n" );
}
/*
* ============================================================================
* Export address book to HTML file.
* Enter: ctl Export control data.
* cache Address book/data source cache.
* Return: Status.
* ============================================================================
*/
void exporthtml_process(
ExportHtmlCtl *ctl, AddressCache *cache )
{
ItemFolder *rootFolder;
FILE *htmlFile;
time_t tt;
gchar *dsName;
static gchar *title;
gchar buf[512];
htmlFile = claws_fopen( ctl->path, "wb" );
if( ! htmlFile ) {
/* Cannot open file */
g_print( "Cannot open file for write\n" );
ctl->retVal = MGU_OPEN_FILE;
return;
}
title = _( "Claws Mail Address Book" );
rootFolder = cache->rootFolder;
dsName = cache->name;
exporthtml_fmt_header( ctl, htmlFile, title );
fprintf( htmlFile, "<body>\n" );
fprintf( htmlFile, "<h1>%s</h1>\n", title );
fprintf( htmlFile, "<p class=\"fmt-folder\">" );
fprintf( htmlFile, "%s: ", _( "Address Book" ) );
fprintf( htmlFile, "%s", dsName );
fprintf( htmlFile, "</p>\n" );
exporthtml_fmt_folder( ctl, htmlFile, rootFolder );
tt = time( NULL );
fprintf( htmlFile, "<p>%s</p>\n", ctime_r( &tt, buf ) );
fprintf( htmlFile, "<hr width=\"100%%\">\n" );
fprintf( htmlFile, "</body>\n" );
fprintf( htmlFile, "</html>\n" );
claws_safe_fclose( htmlFile );
ctl->retVal = MGU_SUCCESS;
/* Create stylesheet files */
exporthtml_create_css_files( ctl );
}
/*
* Build full export file specification.
* Enter: ctl Export control data.
*/
static void exporthtml_build_filespec( ExportHtmlCtl *ctl ) {
gchar *fileSpec;
fileSpec = g_strconcat(
ctl->dirOutput, G_DIR_SEPARATOR_S, ctl->fileHtml, NULL );
ctl->path = mgu_replace_string( ctl->path, fileSpec );
g_free( fileSpec );
}
/*
* ============================================================================
* Parse directory and filename from full export file specification.
* Enter: ctl Export control data.
* fileSpec File spec.
* ============================================================================
*/
void exporthtml_parse_filespec( ExportHtmlCtl *ctl, gchar *fileSpec ) {
gchar *t;
gchar *base = g_path_get_basename(fileSpec);
ctl->fileHtml =
mgu_replace_string( ctl->fileHtml, base );
g_free(base);
t = g_path_get_dirname( fileSpec );
ctl->dirOutput = mgu_replace_string( ctl->dirOutput, t );
g_free( t );
ctl->path = mgu_replace_string( ctl->path, fileSpec );
}
/*
* ============================================================================
* Create output directory.
* Enter: ctl Export control data.
* Return: TRUE if directory created.
* ============================================================================
*/
gboolean exporthtml_create_dir( ExportHtmlCtl *ctl ) {
gboolean retVal = FALSE;
ctl->rcCreate = 0;
if( mkdir( ctl->dirOutput, S_IRWXU ) == 0 ) {
retVal = TRUE;
}
else {
ctl->rcCreate = errno;
}
return retVal;
}
/*
* ============================================================================
* Retrieve create directory error message.
* Enter: ctl Export control data.
* Return: Message.
* ============================================================================
*/
gchar *exporthtml_get_create_msg( ExportHtmlCtl *ctl ) {
gchar *msg;
if( ctl->rcCreate == EEXIST ) {
msg = _( "Name already exists but is not a directory." );
}
else if( ctl->rcCreate == EACCES ) {
msg = _( "No permissions to create directory." );
}
else if( ctl->rcCreate == ENAMETOOLONG ) {
msg = _( "Name is too long." );
}
else {
msg = _( "Not specified." );
}
return msg;
}
/*
* Set default values.
* Enter: ctl Export control data.
*/
static void exporthtml_default_values( ExportHtmlCtl *ctl ) {
gchar *str;
str = g_strconcat(
get_home_dir(), G_DIR_SEPARATOR_S,
DFL_DIR_CLAWS_OUT, NULL );
ctl->dirOutput = mgu_replace_string( ctl->dirOutput, str );
g_free( str );
ctl->fileHtml =
mgu_replace_string( ctl->fileHtml, DFL_FILE_CLAWS_OUT );
ctl->encoding = NULL;
ctl->stylesheet = EXPORT_HTML_ID_DEFAULT;
ctl->nameFormat = EXPORT_HTML_FIRST_LAST;
ctl->banding = TRUE;
ctl->linkEMail = TRUE;
ctl->showAttribs = TRUE;
ctl->retVal = MGU_SUCCESS;
}
/*
* ============================================================================
* Load settings from XML properties file.
* Enter: ctl Export control data.
* ============================================================================
*/
void exporthtml_load_settings( ExportHtmlCtl *ctl ) {
XmlProperty *props;
gint rc;
gchar buf[256];
*buf = '\0';
props = xmlprops_create();
xmlprops_set_path( props, ctl->settingsFile );
rc = xmlprops_load_file( props );
if( rc == 0 ) {
/* Read settings */
xmlprops_get_property_s( props, EXMLPROP_DIRECTORY, buf );
ctl->dirOutput = mgu_replace_string( ctl->dirOutput, buf );
xmlprops_get_property_s( props, EXMLPROP_FILE, buf );
ctl->fileHtml = mgu_replace_string( ctl->fileHtml, buf );
ctl->stylesheet =
xmlprops_get_property_i( props, EXMLPROP_STYLESHEET );
ctl->nameFormat =
xmlprops_get_property_i( props, EXMLPROP_FMT_NAME );
ctl->banding =
xmlprops_get_property_b( props, EXMLPROP_BANDING );
ctl->linkEMail =
xmlprops_get_property_b( props, EXMLPROP_FMT_EMAIL );
ctl->showAttribs =
xmlprops_get_property_b( props, EXMLPROP_FMT_ATTRIB );
}
else {
/* Set default values */
exporthtml_default_values( ctl );
}
exporthtml_build_filespec( ctl );
/* exporthtml_print( ctl, stdout ); */
xmlprops_free( props );
}
/*
* ============================================================================
* Save settings to XML properties file.
* Enter: ctl Export control data.
* ============================================================================
*/
void exporthtml_save_settings( ExportHtmlCtl *ctl ) {
XmlProperty *props;
props = xmlprops_create();
xmlprops_set_path( props, ctl->settingsFile );
xmlprops_set_property( props, EXMLPROP_DIRECTORY, ctl->dirOutput );
xmlprops_set_property( props, EXMLPROP_FILE, ctl->fileHtml );
xmlprops_set_property_i( props, EXMLPROP_STYLESHEET, ctl->stylesheet );
xmlprops_set_property_i( props, EXMLPROP_FMT_NAME, ctl->nameFormat );
xmlprops_set_property_b( props, EXMLPROP_BANDING, ctl->banding );
xmlprops_set_property_b( props, EXMLPROP_FMT_EMAIL, ctl->linkEMail );
xmlprops_set_property_b( props, EXMLPROP_FMT_ATTRIB, ctl->showAttribs );
if (xmlprops_save_file( props ) != MGU_SUCCESS)
g_warning("can't save settings");
xmlprops_free( props );
}
/*
* ============================================================================
* End of Source.
* ============================================================================
*/