mirror of
https://github.com/mozilla/gecko-dev.git
synced 2025-01-16 06:54:00 +00:00
7347 lines
182 KiB
C
7347 lines
182 KiB
C
/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 4 -*-
|
|
*
|
|
* The contents of this file are subject to the Netscape Public License
|
|
* Version 1.0 (the "NPL"); you may not use this file except in
|
|
* compliance with the NPL. You may obtain a copy of the NPL at
|
|
* http://www.mozilla.org/NPL/
|
|
*
|
|
* Software distributed under the NPL is distributed on an "AS IS" basis,
|
|
* WITHOUT WARRANTY OF ANY KIND, either express or implied. See the NPL
|
|
* for the specific language governing rights and limitations under the
|
|
* NPL.
|
|
*
|
|
* The Initial Developer of this code under the NPL is Netscape
|
|
* Communications Corporation. Portions created by Netscape are
|
|
* Copyright (C) 1998 Netscape Communications Corporation. All Rights
|
|
* Reserved.
|
|
*/
|
|
|
|
/*
|
|
editor.c --- XFE editor functions
|
|
Created: Spencer Murray <spence@netscape.com>, 11-Jan-96.
|
|
*/
|
|
|
|
|
|
#include "mozilla.h"
|
|
#include "xfe.h"
|
|
#include "selection.h"
|
|
|
|
#include "net.h"
|
|
|
|
#ifdef EDITOR
|
|
|
|
#include <X11/keysym.h> /* for editor key translation */
|
|
#include <Xm/CutPaste.h> /* for editor key translation */
|
|
|
|
|
|
#include <Xfe/Xfe.h> /* for xfe widgets and utilities */
|
|
|
|
#include <xpgetstr.h> /* for XP_GetString() */
|
|
#include <edttypes.h>
|
|
#include <edt.h>
|
|
#include <layout.h>
|
|
#include <layers.h>
|
|
#include <secnav.h>
|
|
#include <prefapi.h>
|
|
|
|
#include "menu.h"
|
|
#include "fonts.h"
|
|
#include "felocale.h"
|
|
#include "intl_csi.h"
|
|
|
|
#endif /* EDITOR */
|
|
|
|
#include "xeditor.h"
|
|
|
|
#ifdef EDITOR
|
|
|
|
#include "il_icons.h" /* Image icon enumeration. */
|
|
|
|
#include "edtplug.h"
|
|
|
|
#define CB_STATIC static /* let commands.c see it */
|
|
|
|
#define XFE_EDITOR_IM_SUPPORT 1
|
|
|
|
/* Our event loop doesn't have workprocs, so we can't have two modal
|
|
* dialog conditions (one from frontend, one from backend)
|
|
#define EVENT_LOOP_HAS_WORKPROC
|
|
*/
|
|
extern int XFE_EDITOR_NEW_DOCNAME;
|
|
extern int XFE_WARNING_AUTO_SAVE_NEW_MSG;
|
|
extern int XFE_FILE_OPEN;
|
|
extern int XFE_EDITOR_ALERT_FRAME_DOCUMENT;
|
|
extern int XFE_EDITOR_ALERT_ABOUT_DOCUMENT;
|
|
extern int XFE_ALERT_SAVE_CHANGES;
|
|
extern int XFE_WARNING_SAVE_CHANGES;
|
|
extern int XFE_ERROR_GENERIC_ERROR_CODE;
|
|
extern int XFE_EDITOR_COPY_DOCUMENT_BUSY;
|
|
extern int XFE_EDITOR_COPY_SELECTION_EMPTY;
|
|
extern int XFE_EDITOR_COPY_SELECTION_CROSSES_TABLE_DATA_CELL;
|
|
extern int XFE_EDITOR_COPY_DOCUMENT_BUSY;
|
|
extern int XFE_COMMAND_EMPTY;
|
|
extern int XFE_EDITOR_HTML_EDITOR_COMMAND_EMPTY;
|
|
extern int XFE_EDITOR_IMAGE_EDITOR_COMMAND_EMPTY;
|
|
extern int XFE_ACTION_SYNTAX_ERROR;
|
|
extern int XFE_ACTION_WRONG_CONTEXT;
|
|
extern int XFE_EDITOR_WARNING_REMOVE_LINK;
|
|
extern int XFE_UNKNOWN;
|
|
extern int XFE_ERROR_COPYRIGHT_HINT;
|
|
extern int XFE_ERROR_SAVING_OPTIONS;
|
|
extern int XFE_ERROR_READ_ONLY;
|
|
extern int XFE_ERROR_BLOCKED;
|
|
extern int XFE_ERROR_BAD_URL;
|
|
extern int XFE_ERROR_FILE_OPEN;
|
|
extern int XFE_ERROR_FILE_WRITE;
|
|
extern int XFE_ERROR_CREATE_BAKNAME;
|
|
extern int XFE_ERROR_DELETE_BAKFILE;
|
|
extern int XFE_ERROR_WRITING_FILE;
|
|
extern int XFE_ERROR_SRC_NOT_FOUND;
|
|
extern int XFE_WARNING_SAVE_CONTINUE;
|
|
extern int XFE_EDITOR_DOCUMENT_TEMPLATE_EMPTY;
|
|
extern int XFE_EDITOR_BROWSE_LOCATION_EMPTY;
|
|
extern int XFE_UPLOADING_FILE;
|
|
extern int XFE_SAVING_FILE;
|
|
extern int XFE_LOADING_IMAGE_FILE;
|
|
extern int XFE_FILE_N_OF_N;
|
|
extern int XFE_PREPARE_UPLOAD;
|
|
|
|
extern int XP_EDT_SEL_TABLE;
|
|
extern int XP_EDT_SEL_ALL_CELLS;
|
|
extern int XP_EDT_SEL_COL;
|
|
extern int XP_EDT_SEL_ROW;
|
|
extern int XP_EDT_SEL_CELL;
|
|
extern int XP_EDT_SIZE_TABLE_WIDTH;
|
|
extern int XP_EDT_SIZE_TABLE_HEIGHT;
|
|
extern int XP_EDT_SIZE_COL;
|
|
extern int XP_EDT_SIZE_ROW;
|
|
extern int XP_EDT_ADD_ROWS;
|
|
extern int XP_EDT_ADD_COLS;
|
|
extern int XP_EDT_DRAG_TABLE;
|
|
|
|
extern void fe_MailComposeDocumentLoaded(MWContext*);
|
|
extern void fe_HackEditorNotifyToolbarUpdated(MWContext* context);
|
|
#ifdef ENDER
|
|
extern void XFE_EmbeddedEditorViewFocus(MWContext*);
|
|
#endif /* ENDER */
|
|
|
|
static void
|
|
fe_Bell(MWContext* context)
|
|
{
|
|
XBell(XtDisplay(CONTEXT_WIDGET(context)), 0);
|
|
}
|
|
|
|
/*
|
|
* Utility stuff that should be somewhere else.
|
|
*/
|
|
static Widget
|
|
fe_CreateFormDialog(MWContext* context, char* name)
|
|
{
|
|
Widget mainw = CONTEXT_WIDGET(context);
|
|
Widget form;
|
|
Arg av[20];
|
|
int ac;
|
|
Visual* v = 0;
|
|
Colormap cmap = 0;
|
|
Cardinal depth = 0;
|
|
|
|
/*
|
|
* Inherit MainW attributes.
|
|
*/
|
|
XtVaGetValues(mainw, XtNvisual, &v, XtNcolormap, &cmap,
|
|
XtNdepth, &depth, 0);
|
|
ac = 0;
|
|
XtSetArg(av[ac], XmNvisual, v); ac++;
|
|
XtSetArg(av[ac], XmNdepth, depth); ac++;
|
|
XtSetArg(av[ac], XmNcolormap, cmap); ac++;
|
|
XtSetArg(av[ac], XmNtransientFor, mainw); ac++;
|
|
|
|
XtSetArg(av[ac], XmNallowShellResize, TRUE); ac++;
|
|
XtSetArg(av[ac], XmNdialogStyle, XmDIALOG_FULL_APPLICATION_MODAL); ac++;
|
|
XtSetArg(av[ac], XmNautoUnmanage, False); ac++;
|
|
|
|
form = XmCreateFormDialog(mainw, name, av, ac);
|
|
|
|
return form;
|
|
}
|
|
|
|
static int fe_GetStandardPixmap_pixmapsInitialized;
|
|
|
|
static Pixmap
|
|
fe_GetStandardPixmap(MWContext* context, char type)
|
|
{
|
|
Widget mainw = CONTEXT_WIDGET(context);
|
|
Arg av[20];
|
|
int ac;
|
|
Cardinal depth;
|
|
Screen* screen;
|
|
Pixel fg;
|
|
Pixel bg;
|
|
char* name;
|
|
Pixmap pixmap;
|
|
|
|
switch (type) {
|
|
case XmDIALOG_ERROR: name = "xm_error"; break;
|
|
case XmDIALOG_INFORMATION: name = "xm_information"; break;
|
|
case XmDIALOG_QUESTION: name = "xm_question"; break;
|
|
case XmDIALOG_WARNING: name = "xm_warning"; break;
|
|
case XmDIALOG_WORKING: name = "xm_working"; break;
|
|
default:
|
|
return XmUNSPECIFIED_PIXMAP;
|
|
}
|
|
|
|
/*
|
|
* This is so broken. Init MesageBox class, so the Motif
|
|
* standard icons get installed. YUCK..djw
|
|
*/
|
|
if (!fe_GetStandardPixmap_pixmapsInitialized) {
|
|
XtInitializeWidgetClass(xmMessageBoxWidgetClass);
|
|
fe_GetStandardPixmap_pixmapsInitialized++;
|
|
}
|
|
|
|
/*
|
|
* Inherit MainW attributes.
|
|
*/
|
|
ac = 0;
|
|
XtSetArg(av[ac], XmNbackground, &bg); ac++;
|
|
XtSetArg(av[ac], XmNforeground, &fg); ac++;
|
|
XtSetArg(av[ac], XmNdepth, &depth); ac++;
|
|
XtSetArg(av[ac], XmNscreen, &screen); ac++;
|
|
XtGetValues(mainw, av, ac);
|
|
|
|
pixmap = XmGetPixmapByDepth(screen, name, fg, bg, depth);
|
|
if (pixmap == XmUNSPECIFIED_PIXMAP) {
|
|
char default_name[256];
|
|
|
|
strcpy(default_name, "default_");
|
|
strcat(default_name, name);
|
|
|
|
pixmap = XmGetPixmapByDepth(screen, default_name, fg, bg, depth);
|
|
}
|
|
|
|
return pixmap;
|
|
}
|
|
|
|
XtPointer
|
|
fe_GetUserData(Widget widget)
|
|
{
|
|
void* rv;
|
|
XtVaGetValues(widget, XmNuserData, &rv, 0);
|
|
return rv;
|
|
}
|
|
|
|
static struct {
|
|
unsigned type;
|
|
char* name;
|
|
} fe_CreateYesToAllDialog_button_data[] = {
|
|
{ XFE_DIALOG_YES_BUTTON, "yes" },
|
|
{ XFE_DAILOG_YESTOALL_BUTTON, "yesToAll" },
|
|
{ XFE_DIALOG_NO_BUTTON, "no" },
|
|
{ XFE_DIALOG_NOTOALL_BUTTON, "noToAll" },
|
|
{ XFE_DIALOG_CANCEL_BUTTON, "cancel" },
|
|
{ 0, NULL }
|
|
};
|
|
|
|
|
|
|
|
#define XFE_YESTOALL_YES_BUTTON (fe_CreateYesToAllDialog_button_data[0].name)
|
|
#define XFE_YESTOALL_YESTOALL_BUTTON (fe_CreateYesToAllDialog_button_data[1].name)
|
|
#define XFE_YESTOALL_NO_BUTTON (fe_CreateYesToAllDialog_button_data[2].name)
|
|
#define XFE_YESTOALL_NOTOALL_BUTTON (fe_CreateYesToAllDialog_button_data[3].name)
|
|
#define XFE_YESTOALL_CANCEL_BUTTON (fe_CreateYesToAllDialog_button_data[4].name)
|
|
|
|
typedef struct {
|
|
Widget widget;
|
|
char* name;
|
|
} GetChildInfo;
|
|
|
|
static XtPointer
|
|
fe_YesToAllDialogGetChildMappee(Widget widget, XtPointer data)
|
|
{
|
|
GetChildInfo* info = (GetChildInfo*)data;
|
|
char* name = XtName(widget);
|
|
|
|
if (strcmp(info->name, name) == 0) {
|
|
info->widget = widget;
|
|
return (XtPointer)1; /* don't look any more */
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
Widget
|
|
fe_YesToAllDialogGetChild(Widget parent, unsigned char type)
|
|
{
|
|
char* name;
|
|
GetChildInfo info;
|
|
|
|
switch (type) {
|
|
case XFE_DIALOG_YES_BUTTON:
|
|
name = XFE_YESTOALL_YES_BUTTON;
|
|
break;
|
|
case XFE_DAILOG_YESTOALL_BUTTON:
|
|
name = XFE_YESTOALL_YESTOALL_BUTTON;
|
|
break;
|
|
case XFE_DIALOG_NO_BUTTON:
|
|
name = XFE_YESTOALL_NO_BUTTON;
|
|
break;
|
|
case XFE_DIALOG_NOTOALL_BUTTON:
|
|
name = XFE_YESTOALL_NOTOALL_BUTTON;
|
|
break;
|
|
case XFE_DIALOG_CANCEL_BUTTON:
|
|
name = XFE_YESTOALL_CANCEL_BUTTON;
|
|
break;
|
|
default:
|
|
return NULL; /* kill, kill, kill */
|
|
}
|
|
|
|
info.name = name;
|
|
info.widget = NULL;
|
|
#ifdef OSF1
|
|
fe_WidgetTreeWalk(parent, fe_YesToAllDialogGetChildMappee, (void *)&info);
|
|
#else
|
|
fe_WidgetTreeWalk(parent, fe_YesToAllDialogGetChildMappee, &info);
|
|
#endif
|
|
|
|
return info.widget;
|
|
}
|
|
|
|
static Widget
|
|
fe_CreateYesToAllDialog(MWContext* context, char* name, Arg* p_av, Cardinal p_ac)
|
|
{
|
|
Arg av[20];
|
|
Cardinal ac;
|
|
Widget form;
|
|
Widget children[16];
|
|
Cardinal nchildren = 0;
|
|
Pixmap pixmap;
|
|
char* bname;
|
|
Widget button;
|
|
Widget icon;
|
|
Widget text;
|
|
Widget separator;
|
|
Widget row;
|
|
int i;
|
|
XtCallbackRec* button_callback_rec = NULL;
|
|
XmString msg_string = NULL;
|
|
char namebuf[64];
|
|
char pixmapType = XmDIALOG_WARNING;
|
|
|
|
for (i = 0; i < p_ac; i++) {
|
|
if (p_av[i].name == XmNarmCallback)
|
|
button_callback_rec = (XtCallbackRec*)p_av[i].value;
|
|
else if (p_av[i].name == XmNmessageString)
|
|
msg_string = (XmString)p_av[i].value;
|
|
else if (p_av[i].name == XmNdialogType)
|
|
pixmapType = (unsigned char)p_av[i].value;
|
|
}
|
|
|
|
form = fe_CreateFormDialog(context, name);
|
|
|
|
strcpy(namebuf, name); strcat(namebuf, "Message");
|
|
ac = 0;
|
|
XtSetArg(av[ac], XmNtopAttachment, XmATTACH_FORM); ac++;
|
|
XtSetArg(av[ac], XmNleftAttachment, XmATTACH_FORM); ac++;
|
|
XtSetArg(av[ac], XmNrightAttachment, XmATTACH_FORM); ac++;
|
|
XtSetArg(av[ac], XmNorientation, XmHORIZONTAL); ac++;
|
|
XtSetArg(av[ac], XmNentryAlignment, XmALIGNMENT_CENTER); ac++;
|
|
XtSetArg(av[ac], XmNisAligned, TRUE); ac++;
|
|
XtSetArg(av[ac], XmNpacking, XmPACK_TIGHT); ac++;
|
|
row = XmCreateRowColumn(form, namebuf, av, ac);
|
|
|
|
nchildren = 0;
|
|
|
|
/*
|
|
* Make pixmap label.
|
|
*/
|
|
pixmap = fe_GetStandardPixmap(context, pixmapType);
|
|
|
|
ac = 0;
|
|
XtSetArg(av[ac], XmNlabelType, XmPIXMAP); ac++;
|
|
XtSetArg(av[ac], XmNlabelPixmap, pixmap); ac++;
|
|
icon = XmCreateLabelGadget(row, "icon", av, ac);
|
|
children[nchildren++] = icon;
|
|
|
|
/*
|
|
* Text.
|
|
*/
|
|
ac = 0;
|
|
if (msg_string)
|
|
XtSetArg(av[ac], XmNlabelString, msg_string); ac++;
|
|
XtSetArg(av[ac], XmNalignment, XmALIGNMENT_BEGINNING); ac++;
|
|
XtSetArg(av[ac], XmNlabelType, XmSTRING); ac++;
|
|
text = XmCreateLabelGadget(row, "text", av, ac);
|
|
children[nchildren++] = text;
|
|
|
|
XtManageChildren(children, nchildren); /* children of row */
|
|
|
|
nchildren = 0;
|
|
children[nchildren++] = row;
|
|
|
|
/*
|
|
* Separator.
|
|
*/
|
|
ac = 0;
|
|
XtSetArg(av[ac], XmNtopAttachment, XmATTACH_WIDGET); ac++;
|
|
XtSetArg(av[ac], XmNtopWidget, row); ac++;
|
|
XtSetArg(av[ac], XmNleftAttachment, XmATTACH_FORM); ac++;
|
|
XtSetArg(av[ac], XmNrightAttachment, XmATTACH_FORM); ac++;
|
|
separator = XmCreateSeparatorGadget(form, "separator", av, ac);
|
|
children[nchildren++] = separator;
|
|
|
|
/*
|
|
* Yes, YesToAll, No, NoToAll, Cancel.
|
|
*/
|
|
strcpy(namebuf, name); strcat(namebuf, "Buttons");
|
|
ac = 0;
|
|
XtSetArg(av[ac], XmNtopAttachment, XmATTACH_WIDGET); ac++;
|
|
XtSetArg(av[ac], XmNtopWidget, separator); ac++;
|
|
XtSetArg(av[ac], XmNleftAttachment, XmATTACH_FORM); ac++;
|
|
XtSetArg(av[ac], XmNrightAttachment, XmATTACH_FORM); ac++;
|
|
XtSetArg(av[ac], XmNbottomAttachment, XmATTACH_FORM); ac++;
|
|
XtSetArg(av[ac], XmNorientation, XmHORIZONTAL); ac++;
|
|
XtSetArg(av[ac], XmNentryAlignment, XmALIGNMENT_CENTER); ac++;
|
|
XtSetArg(av[ac], XmNisAligned, TRUE); ac++;
|
|
XtSetArg(av[ac], XmNpacking, XmPACK_COLUMN); ac++;
|
|
row = XmCreateRowColumn(form, namebuf, av, ac);
|
|
children[nchildren++] = row;
|
|
|
|
XtManageChildren(children, nchildren);
|
|
|
|
nchildren = 0; /* now do buttons */
|
|
button = NULL;
|
|
for (i = 0; (bname = fe_CreateYesToAllDialog_button_data[i].name); i++) {
|
|
|
|
ac = 0;
|
|
XtSetArg(av[ac], XmNalignment, XmALIGNMENT_CENTER); ac++;
|
|
XtSetArg(av[ac], XmNlabelType, XmSTRING); ac++;
|
|
button = XmCreatePushButtonGadget(row, bname, av, ac);
|
|
|
|
if (button_callback_rec)
|
|
XtAddCallback(
|
|
button,
|
|
XmNactivateCallback,
|
|
button_callback_rec->callback,
|
|
button_callback_rec->closure
|
|
);
|
|
children[nchildren++] = button;
|
|
}
|
|
|
|
XtManageChildren(children, nchildren);
|
|
|
|
return form;
|
|
}
|
|
|
|
Boolean
|
|
fe_EditorDocumentIsSaved(MWContext* context)
|
|
{
|
|
History_entry* hist_ent;
|
|
|
|
if (!context)
|
|
return False;
|
|
|
|
if (EDT_IS_NEW_DOCUMENT(context))
|
|
return False;
|
|
|
|
hist_ent = SHIST_GetCurrent(&context->hist);
|
|
|
|
if (!hist_ent)
|
|
return False;
|
|
|
|
if (hist_ent->address != NULL && NET_IsLocalFileURL(hist_ent->address))
|
|
return True;
|
|
else
|
|
return False;
|
|
}
|
|
|
|
|
|
Boolean fe_SaveAsDialog(MWContext* context, char* buf, int type);
|
|
|
|
static Boolean
|
|
fe_editor_report_file_error(MWContext* context,
|
|
ED_FileError code,
|
|
char* filename,
|
|
Boolean ask_question)
|
|
{
|
|
int id;
|
|
char* msg;
|
|
char buf[MAXPATHLEN];
|
|
|
|
switch (code) {
|
|
case ED_ERROR_READ_ONLY: /* File is marked read-only */
|
|
id = XFE_ERROR_READ_ONLY;
|
|
break;
|
|
case ED_ERROR_BLOCKED: /* Can't write at this time, edit buffer blocked */
|
|
id = XFE_ERROR_BLOCKED;
|
|
break;
|
|
case ED_ERROR_BAD_URL: /* URL was not a "file:" type or no string */
|
|
id = XFE_ERROR_BAD_URL;
|
|
break;
|
|
case ED_ERROR_FILE_OPEN:
|
|
id = XFE_ERROR_FILE_OPEN;
|
|
break;
|
|
case ED_ERROR_FILE_WRITE:
|
|
id = XFE_ERROR_FILE_WRITE;
|
|
break;
|
|
case ED_ERROR_CREATE_BAKNAME:
|
|
case ED_ERROR_FILE_RENAME_TO_BAK:
|
|
id = XFE_ERROR_CREATE_BAKNAME;
|
|
break;
|
|
case ED_ERROR_DELETE_BAKFILE:
|
|
id = XFE_ERROR_DELETE_BAKFILE;
|
|
break;
|
|
case ED_ERROR_SRC_NOT_FOUND:
|
|
id = XFE_ERROR_SRC_NOT_FOUND;
|
|
break;
|
|
default:
|
|
id = XFE_ERROR_GENERIC_ERROR_CODE; /* generic "...code = (%d)" */
|
|
break;
|
|
}
|
|
|
|
msg = XP_GetString(XFE_ERROR_WRITING_FILE);
|
|
sprintf(buf, msg, filename);
|
|
strcat(buf, "\n\n");
|
|
|
|
msg = XP_GetString(id);
|
|
sprintf(&buf[strlen(buf)], msg, code);
|
|
|
|
if (ask_question) {
|
|
strcat(buf, "\n\n");
|
|
strcat(buf, XP_GetString(XFE_WARNING_SAVE_CONTINUE));
|
|
return XFE_Confirm(context, buf);
|
|
} else {
|
|
FE_Alert(context, buf);
|
|
return TRUE;
|
|
}
|
|
}
|
|
|
|
static Boolean
|
|
fe_editor_save_common(MWContext* context, char* target_url)
|
|
{
|
|
History_entry* hist_ent;
|
|
ED_FileError code;
|
|
Boolean images;
|
|
Boolean links;
|
|
Boolean save_as;
|
|
|
|
/* url->address is used for filename */
|
|
hist_ent = SHIST_GetCurrent(&context->hist);
|
|
|
|
if (!hist_ent)
|
|
return FALSE;
|
|
|
|
/*
|
|
* Has the title changed since it was last saved?
|
|
*/
|
|
if (hist_ent->title && strcmp(hist_ent->title, hist_ent->address) == 0) {
|
|
/*
|
|
* BE will set this to the new file, if we don't zap it,
|
|
* BE will retain the old name - yuck!
|
|
*/
|
|
XP_FREE(hist_ent->title);
|
|
hist_ent->title = NULL;
|
|
}
|
|
|
|
if (target_url) { /* save as */
|
|
save_as = TRUE;
|
|
/* fe_EditorPreferencesGetLinksAndImages(context, &links, &images); */
|
|
} else { /* save me */
|
|
target_url = hist_ent->address;
|
|
save_as = FALSE;
|
|
/* images = FALSE;
|
|
links = FALSE; */
|
|
}
|
|
/* hardts, now always use prefs for links and images, Save and Save As
|
|
do the same thing for adjusting links and images */
|
|
fe_EditorPreferencesGetLinksAndImages(context, &links, &images);
|
|
|
|
/*
|
|
* Do this before the save starts, as save very quickly gets
|
|
* asynchronous.
|
|
*/
|
|
CONTEXT_DATA(context)->is_file_saving = True;
|
|
|
|
/*
|
|
* EDT_SaveFile returns true if it completed immediately.
|
|
* savesas uses source and dest, save just uses
|
|
*/
|
|
code = EDT_SaveFile(context, hist_ent->address, target_url,
|
|
save_as, images, links);
|
|
|
|
/* here we spin and spin until all the savings of images
|
|
* and stuff are done so that we get a "real" status back from
|
|
* the backend. This makes saving semi-synchronous, good for
|
|
* status reporting
|
|
*/
|
|
if (code == ED_ERROR_NONE) { /* we didn't error straight away */
|
|
while(CONTEXT_DATA(context)->is_file_saving) {
|
|
/* do nothing, we wait here */
|
|
fe_EventLoop();
|
|
}
|
|
}
|
|
|
|
code = CONTEXT_DATA(context)->file_save_status;
|
|
if (code != ED_ERROR_NONE) {
|
|
/*
|
|
* I'm sure if it is bad if the title is NULL, but
|
|
* just in case..
|
|
*/
|
|
if ((hist_ent = SHIST_GetCurrent(&context->hist)) != NULL
|
|
&&
|
|
hist_ent->title == NULL) {
|
|
hist_ent->title = XP_STRDUP(hist_ent->address);
|
|
}
|
|
fe_editor_report_file_error(context, code, target_url, FALSE);
|
|
return FALSE;
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
Boolean
|
|
fe_EditorSaveAs(MWContext* context) /* launches dialog */
|
|
{
|
|
char filebuf[MAXPATHLEN];
|
|
|
|
if (fe_SaveAsDialog(context, filebuf, fe_FILE_TYPE_HTML)) {
|
|
|
|
char urlbuf[MAXPATHLEN];
|
|
|
|
PR_snprintf(urlbuf, sizeof(urlbuf), "file:%.900s", filebuf);
|
|
|
|
return fe_editor_save_common(context, urlbuf);
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
Boolean
|
|
fe_EditorSave(MWContext* context)
|
|
{
|
|
/*
|
|
* Always ask the user for the filename once.
|
|
*
|
|
* if (EDT_IS_NEW_DOCUMENT(context))
|
|
*/
|
|
if (!fe_EditorDocumentIsSaved(context)) {
|
|
return fe_EditorSaveAs(context);
|
|
}
|
|
|
|
return fe_editor_save_common(context, NULL);
|
|
}
|
|
|
|
/*
|
|
* I18N input method support.
|
|
*/
|
|
#ifdef XFE_EDITOR_IM_SUPPORT
|
|
static void
|
|
fe_editor_im_deregister(Widget widget, XtPointer closure, XtPointer cb_data)
|
|
{
|
|
if (!fe_globalData.editor_im_input_enabled)
|
|
return;
|
|
|
|
XmImUnregister(widget);
|
|
}
|
|
|
|
static void
|
|
fe_editor_im_get_coords(MWContext* context, XPoint* point)
|
|
{
|
|
fe_EditorCaretData* data = &EDITOR_CONTEXT_DATA(context)->caret_data;
|
|
|
|
if (!fe_globalData.editor_im_input_enabled)
|
|
return;
|
|
|
|
if (data->running == FALSE && data->showing == FALSE) {
|
|
point->x = 0;
|
|
point->y = 16;
|
|
} else {
|
|
point->x = data->x + data->width - CONTEXT_DATA(context)->document_x;
|
|
point->y = data->y - 2 - CONTEXT_DATA(context)->document_y +
|
|
data->height;
|
|
}
|
|
#ifdef DEBUG_rhess2
|
|
fprintf(stderr, "im_point::[ %d, %d ]\n", point->x, point->y);
|
|
#endif
|
|
}
|
|
|
|
static void
|
|
fe_editor_im_init(MWContext* context)
|
|
{
|
|
Widget widget = CONTEXT_DATA(context)->drawing_area;
|
|
Arg args[8];
|
|
Cardinal n;
|
|
XmFontList font_list;
|
|
Pixel bg_pixel;
|
|
Pixel fg_pixel;
|
|
/* Pixmap bg_pixmap; */
|
|
XPoint xmim_point;
|
|
/* int xmim_height; */
|
|
|
|
if (!fe_globalData.editor_im_input_enabled)
|
|
#ifdef DEBUG_rhess
|
|
{
|
|
fprintf(stderr, "fe_editor_im_init::[ PUNT ]\n");
|
|
}
|
|
else
|
|
{
|
|
fprintf(stderr, "fe_editor_im_init::[ ]\n");
|
|
}
|
|
#else
|
|
return;
|
|
#endif
|
|
|
|
XtVaSetValues(CONTEXT_WIDGET(context), XmNallowShellResize, TRUE, NULL);
|
|
|
|
XmImRegister(widget, 0);
|
|
XtAddCallback(widget, XmNdestroyCallback, fe_editor_im_deregister, NULL);
|
|
|
|
/*
|
|
* Should these change dynamically with point?
|
|
*/
|
|
#ifdef OSF1
|
|
font_list = (void *)fe_GetFont(context, 3, LO_FONT_FIXED);
|
|
#else
|
|
font_list = fe_GetFont(context, 3, LO_FONT_FIXED);
|
|
#endif
|
|
bg_pixel = CONTEXT_DATA(context)->bg_pixel;
|
|
fg_pixel = CONTEXT_DATA(context)->fg_pixel;
|
|
/* bg_pixmap = CONTEXT_DATA(context)->backdrop_pixmap; */
|
|
|
|
fe_editor_im_get_coords(context, &xmim_point);
|
|
|
|
n = 0;
|
|
XtSetArg(args[n], XmNfontList, font_list); n++;
|
|
XtSetArg(args[n], XmNbackground, bg_pixel); n++;
|
|
XtSetArg(args[n], XmNforeground, fg_pixel); n++;
|
|
/* XtSetArg(args[n], XmNbackgroundPixmap, bg_pixmap); n++; */
|
|
XtSetArg(args[n], XmNspotLocation, &xmim_point); n++;
|
|
/* XtSetArg(args[n], XmNlineSpace, xmim_height); n++; */
|
|
XmImSetValues(widget, args, n);
|
|
}
|
|
|
|
static void
|
|
fe_editor_im_set_coords(MWContext* context)
|
|
{
|
|
Widget widget = CONTEXT_DATA(context)->drawing_area;
|
|
Arg args[8];
|
|
Cardinal n;
|
|
XPoint xmim_point;
|
|
/* int xmim_height; */
|
|
|
|
if (!fe_globalData.editor_im_input_enabled)
|
|
return;
|
|
|
|
fe_editor_im_get_coords(context, &xmim_point);
|
|
|
|
n = 0;
|
|
XtSetArg(args[n], XmNspotLocation, &xmim_point); n++;
|
|
/* XtSetArg(args[n], XmNlineSpace, xmim_height); n++; */
|
|
XmImSetFocusValues(widget, args, n);
|
|
}
|
|
|
|
static void
|
|
fe_editor_im_focus_in(MWContext* context)
|
|
{
|
|
if (!fe_globalData.editor_im_input_enabled)
|
|
return;
|
|
|
|
fe_editor_im_set_coords(context);
|
|
|
|
XtSetKeyboardFocus(CONTEXT_WIDGET(context),
|
|
CONTEXT_DATA(context)->drawing_area);
|
|
}
|
|
|
|
static void
|
|
fe_editor_im_focus_out(MWContext* context)
|
|
{
|
|
Widget widget = CONTEXT_DATA(context)->drawing_area;
|
|
|
|
if (!fe_globalData.editor_im_input_enabled)
|
|
return;
|
|
|
|
XmImUnsetFocus(widget);
|
|
}
|
|
#endif /*XFE_EDITOR_IM_SUPPORT*/
|
|
|
|
static void fe_caret_pause(MWContext* context);
|
|
static void fe_caret_unpause(MWContext* context);
|
|
|
|
void
|
|
xfe2_EditorImSetCoords(MWContext* context)
|
|
{
|
|
fe_editor_im_set_coords(context);
|
|
}
|
|
|
|
void
|
|
xfe2_EditorCaretShow(MWContext* context)
|
|
{
|
|
fe_caret_unpause(context);
|
|
}
|
|
|
|
void
|
|
xfe2_EditorCaretHide(MWContext* context)
|
|
{
|
|
fe_caret_pause(context);
|
|
}
|
|
|
|
void
|
|
xfe2_EditorWaitWhileContextBusy(MWContext* context)
|
|
{
|
|
int n = 0;
|
|
|
|
while (XP_IsContextBusy(context)) {
|
|
NET_ProcessNet(0, NET_EVERYTIME_TYPE);
|
|
#ifdef MOZ_MAIL_NEWS
|
|
MSG_OnIdle();
|
|
#endif
|
|
#ifdef DEBUG_rhess
|
|
fprintf(stderr, "tic::[ %d ]\n", n);
|
|
#endif
|
|
n++;
|
|
}
|
|
}
|
|
|
|
static void
|
|
fe_change_focus_eh(Widget w, XtPointer closure, XEvent* event, Boolean* cont)
|
|
{
|
|
MWContext* context = (MWContext *)closure;
|
|
|
|
*cont = TRUE;
|
|
|
|
if (event->type == FocusIn)
|
|
{
|
|
fe_caret_unpause(context);
|
|
#ifdef XFE_EDITOR_IM_SUPPORT
|
|
fe_editor_im_focus_in(context);
|
|
#else
|
|
XtSetKeyboardFocus(CONTEXT_WIDGET(context),
|
|
CONTEXT_DATA(context)->drawing_area);
|
|
|
|
#endif /*XFE_EDITOR_IM_SUPPORT*/
|
|
}
|
|
else
|
|
{
|
|
fe_caret_pause(context);
|
|
#ifdef XFE_EDITOR_IM_SUPPORT
|
|
fe_editor_im_focus_out(context);
|
|
#else
|
|
XtSetKeyboardFocus(CONTEXT_WIDGET(context), NULL);
|
|
#endif /*XFE_EDITOR_IM_SUPPORT*/
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Menu stuff.
|
|
*/
|
|
typedef MWContext MWEditorContext; /* place holder */
|
|
|
|
/*
|
|
* NOTE: look in EditorFrame.cpp for new implementation...
|
|
*/
|
|
|
|
void
|
|
xfe2_EditorInit(MWContext* context)
|
|
{
|
|
fe_EditorCaretData* data = &EDITOR_CONTEXT_DATA(context)->caret_data;
|
|
#if 0
|
|
/*
|
|
* Just cannot get this to do what it is supposed to.
|
|
*/
|
|
XtAddEventHandler(CONTEXT_WIDGET(context), PropertyChangeMask,
|
|
FALSE, (XtEventHandler)fe_property_notify_eh,
|
|
context);
|
|
#endif
|
|
|
|
if (context->type == MWContextEditor) {
|
|
Widget w;
|
|
|
|
/*
|
|
* NOTE: the MWContextMessageComposition has it's own ev handler...
|
|
*
|
|
*/
|
|
#ifdef DEBUG_rhess
|
|
fprintf(stderr, "tag1::[ %p ]\n", CONTEXT_WIDGET(context));
|
|
#endif
|
|
#ifdef ENDER
|
|
if (EDITOR_CONTEXT_DATA(context)->embedded)
|
|
w = CONTEXT_DATA(context)->drawing_area;
|
|
else
|
|
#endif /* ENDER */
|
|
w = CONTEXT_WIDGET(context);
|
|
|
|
XtAddEventHandler(w, FocusChangeMask,
|
|
FALSE, (XtEventHandler)fe_change_focus_eh,
|
|
context);
|
|
}
|
|
|
|
/*
|
|
* Register with input server.
|
|
*/
|
|
#ifdef XFE_EDITOR_IM_SUPPORT
|
|
fe_editor_im_init(context);
|
|
#endif /*XFE_EDITOR_IM_SUPPORT*/
|
|
|
|
/*
|
|
* NOTE: initialize the caret data structure... [ may be overkill ]
|
|
*/
|
|
data->x = 0;
|
|
data->y = 0;
|
|
data->width = 0;
|
|
data->height = 0;
|
|
data->time = 0;
|
|
data->timer_id = 0;
|
|
data->showing = FALSE;
|
|
data->running = FALSE;
|
|
data->backing_store = 0;
|
|
}
|
|
|
|
char*
|
|
fe_EditorGetTemplateURL(MWContext* context)
|
|
{
|
|
char* template_url = fe_EditorPreferencesGetTemplate(context);
|
|
|
|
if (template_url == NULL || template_url[0] == '\0') {
|
|
char* msg = XP_GetString(XFE_EDITOR_DOCUMENT_TEMPLATE_EMPTY);
|
|
|
|
/* The new document template location is not set. */
|
|
if (XFE_Confirm(context, msg)) {
|
|
fe_EditorPreferencesDialogDo(context,
|
|
XFE_EDITOR_DOCUMENT_PROPERTIES_GENERAL);
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
return template_url;
|
|
}
|
|
|
|
char*
|
|
fe_EditorGetWizardURL(MWContext* context)
|
|
{
|
|
return XFE_WIZARD_TEMPLATE_URL;
|
|
}
|
|
|
|
void
|
|
fe_EditorDisplaySource(MWContext* context)
|
|
{
|
|
EDT_DisplaySource(context);
|
|
}
|
|
|
|
void
|
|
fe_NukeCaret(MWContext * context)
|
|
{
|
|
fe_EditorCaretData* data = &EDITOR_CONTEXT_DATA(context)->caret_data;
|
|
|
|
#if DEBUG_rhess2
|
|
fprintf(stderr, "fe_NukeCaret::[ ]\n");
|
|
#endif
|
|
|
|
data->x = 0;
|
|
data->y = 0;
|
|
data->width = 0;
|
|
data->height = 0;
|
|
data->time = 0;
|
|
data->timer_id = 0;
|
|
data->showing = False;
|
|
data->running = False;
|
|
data->backing_store = 0;
|
|
}
|
|
|
|
void
|
|
fe_EditorCleanup(MWContext* context)
|
|
{
|
|
Boolean as_enabled;
|
|
unsigned as_time;
|
|
|
|
/*
|
|
* If they have autosave on, try to do a save. Don't do it
|
|
* for a new doc, as that'll mean dialogs, and .....
|
|
*/
|
|
if (context->type != MWContextMessageComposition &&
|
|
!EDT_IS_NEW_DOCUMENT(context) &&
|
|
#ifdef ENDER
|
|
!EDITOR_CONTEXT_DATA(context)->embedded &&
|
|
#endif /* ENDER */
|
|
(!EDT_IsBlocked(context) && EDT_DirtyFlag(context))) {
|
|
fe_EditorPreferencesGetAutoSave(context, &as_enabled, &as_time);
|
|
if (as_enabled)
|
|
fe_EditorSave(context);
|
|
}
|
|
|
|
fe_NukeCaret(context);
|
|
EDT_DestroyEditBuffer(context);
|
|
}
|
|
|
|
char*
|
|
fe_EditorMakeName(MWContext* context, char* buf, unsigned maxsize)
|
|
{
|
|
History_entry* entry;
|
|
char* local_name = NULL;
|
|
|
|
if (!EDT_IS_NEW_DOCUMENT(context)
|
|
&&
|
|
(entry = SHIST_GetCurrent(&context->hist)) && (entry->address)) {
|
|
|
|
local_name = NULL;
|
|
if (XP_ConvertUrlToLocalFile(entry->address, &local_name)) {
|
|
FE_CondenseURL(buf, local_name, maxsize);
|
|
XP_FREE(local_name);
|
|
} else {
|
|
FE_CondenseURL(buf, entry->address, maxsize);
|
|
}
|
|
} else {
|
|
strcpy(buf, XP_GetString(XFE_EDITOR_NEW_DOCNAME));
|
|
}
|
|
return buf;
|
|
}
|
|
|
|
static Boolean
|
|
fe_named_question(MWContext* context, char* name, char* message)
|
|
{
|
|
return (Boolean)((int)fe_dialog(CONTEXT_WIDGET (context),
|
|
name, message,
|
|
TRUE, 0, TRUE, FALSE, 0));
|
|
}
|
|
|
|
Boolean
|
|
fe_save_file_check(MWContext* context, Boolean file_must_exist, Boolean autoSaveNew)
|
|
{
|
|
int rv;
|
|
char filename[MAXPATHLEN];
|
|
char buf[MAXPATHLEN];
|
|
char dialogmessages[MAXPATHLEN+64];
|
|
Boolean ok_cancel = (file_must_exist && EDT_IS_NEW_DOCUMENT(context));
|
|
|
|
if (ok_cancel || EDT_DirtyFlag(context)) {
|
|
fe_EditorMakeName(context, filename, sizeof(filename));
|
|
|
|
if (ok_cancel) {
|
|
/* You must save <filename> as a local file */
|
|
sprintf(buf, XP_GetString(XFE_ALERT_SAVE_CHANGES), filename);
|
|
rv = (fe_named_question(context, "saveNewFile", buf))?
|
|
XmDIALOG_OK_BUTTON: XmDIALOG_CANCEL_BUTTON;
|
|
} else {
|
|
/* Save changes to <filename>? */
|
|
sprintf(buf, XP_GetString(XFE_WARNING_SAVE_CHANGES), filename);
|
|
if (autoSaveNew) {
|
|
sprintf(dialogmessages,"%s\n\n%s", buf,
|
|
XP_GetString(XFE_WARNING_AUTO_SAVE_NEW_MSG));
|
|
rv = fe_YesNoCancelDialog(context, "autoSaveNew", dialogmessages);
|
|
} else
|
|
rv = fe_YesNoCancelDialog(context, "saveFile", buf);
|
|
}
|
|
|
|
if (rv == XmDIALOG_OK_BUTTON)
|
|
return fe_EditorSave(context);
|
|
else if (rv == XmDIALOG_CANCEL_BUTTON)
|
|
return FALSE;
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
XP_Bool
|
|
FE_CheckAndSaveDocument(MWContext* context)
|
|
{
|
|
return fe_save_file_check(context, TRUE, FALSE);
|
|
}
|
|
|
|
XP_Bool
|
|
FE_CheckAndAutoSaveDocument(MWContext *context)
|
|
{
|
|
return fe_save_file_check(context, FALSE, TRUE);
|
|
}
|
|
|
|
/*
|
|
* NOTE: this routine is currently commented out, always returns TRUE.
|
|
* Call this to check if we need to force saving file
|
|
* Conditions are New, unsaved document and when editing a remote file
|
|
* Returns TRUE for all cases except CANCEL by the user in any dialog
|
|
*/
|
|
XP_Bool
|
|
FE_SaveNonLocalDocument(MWContext* context, XP_Bool save_new_document)
|
|
{
|
|
#if 0
|
|
History_entry* hist_entry;
|
|
int type;
|
|
Boolean links;
|
|
Boolean images;
|
|
Boolean save;
|
|
char filebuf[MAXPATHLEN];
|
|
char urlbuf[MAXPATHLEN];
|
|
ED_FileError file_error;
|
|
|
|
if (context == NULL || !EDT_IS_EDITOR(context)) {
|
|
return TRUE;
|
|
}
|
|
|
|
hist_entry = SHIST_GetCurrent(&(context->hist));
|
|
if (!hist_entry || !hist_entry->address)
|
|
return TRUE;
|
|
|
|
/*
|
|
*/
|
|
type = NET_URL_Type(hist_entry->address);
|
|
|
|
if ((type > 0 && type != FILE_TYPE_URL && type != FILE_CACHE_TYPE_URL &&
|
|
type != MAILBOX_TYPE_URL && type != VIEW_SOURCE_TYPE_URL)
|
|
||
|
|
(save_new_document && EDT_IS_NEW_DOCUMENT(context)))
|
|
{
|
|
fe_EditorPreferencesGetLinksAndImages(context, &links, &images);
|
|
fe_DoSaveRemoteDialog(context, &save, &links, &images);
|
|
|
|
if (!save)
|
|
return FALSE;
|
|
|
|
#if 0
|
|
if (!EDT_IS_NEW_DOCUMENT(context))
|
|
fe_EditorCopyrightWarningDialogDo();
|
|
#endif
|
|
|
|
if (!fe_SaveAsDialog(context, filebuf, fe_FILE_TYPE_HTML))
|
|
return FALSE;
|
|
|
|
/*
|
|
* Grab the filename while we have it to form file: path.
|
|
*/
|
|
PR_snprintf(urlbuf, sizeof(urlbuf), "file:%.900s", filebuf);
|
|
|
|
/* the c++ comments are here deliberately so that later
|
|
* on if we use this code we won't forget to modify it */
|
|
|
|
/* this here is all disabled, we don't use it anymore but if later
|
|
* on we need to use it again, we should probably make it synchronous
|
|
*/
|
|
|
|
file_error = EDT_SaveFile(context,
|
|
hist_entry->address,
|
|
urlbuf,
|
|
TRUE,
|
|
images,
|
|
links);
|
|
|
|
if (file_error != ED_ERROR_NONE) {
|
|
fe_editor_report_file_error(context, file_error, urlbuf,
|
|
FALSE);
|
|
return FALSE;
|
|
}
|
|
}
|
|
#endif
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
CB_STATIC void
|
|
fe_editor_delete_cb(Widget widget, XtPointer closure, XtPointer call_data)
|
|
{
|
|
MWContext *context = (MWContext *)closure;
|
|
|
|
fe_UserActivity (context);
|
|
|
|
if (fe_WindowCount == 1) {
|
|
fe_QuitCallback(widget, closure, call_data);
|
|
} else {
|
|
if (fe_save_file_check(context, FALSE, FALSE))
|
|
fe_EditorDelete(context);
|
|
}
|
|
}
|
|
|
|
void fe_editor_delete_response(Widget widget, XtPointer closure, XtPointer call_data)
|
|
{
|
|
fe_editor_delete_cb(widget, closure, call_data);
|
|
}
|
|
|
|
|
|
Boolean
|
|
fe_EditorCheckUnsaved(MWContext* context)
|
|
{
|
|
struct fe_MWContext_cons* rest;
|
|
|
|
for (rest = fe_all_MWContexts; rest; rest = rest->next) {
|
|
MWContext* context = rest->context;
|
|
if (context->type == MWContextEditor) {
|
|
if (!fe_save_file_check(context, FALSE, FALSE)) {
|
|
return FALSE;
|
|
}
|
|
}
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
void
|
|
fe_EditorReload(MWContext* context, Boolean super_reload)
|
|
{
|
|
if (EDT_IS_NEW_DOCUMENT(context))
|
|
return; /* only from action */
|
|
|
|
if (!FE_CheckAndSaveDocument(context))
|
|
return;
|
|
|
|
#if 0 /* this does not exist! */
|
|
EDT_Reload(context);
|
|
#else
|
|
if (super_reload)
|
|
fe_ReLayout (context, NET_SUPER_RELOAD);
|
|
else
|
|
fe_ReLayout (context, NET_NORMAL_RELOAD);
|
|
#endif
|
|
}
|
|
|
|
void
|
|
fe_EditorFind(MWContext* context)
|
|
{
|
|
MWContext* top_context = XP_GetNonGridContext(context);
|
|
|
|
if (!top_context)
|
|
top_context = context;
|
|
|
|
fe_UserActivity(top_context);
|
|
|
|
fe_FindAndReplaceDialog(top_context, False);
|
|
}
|
|
|
|
Boolean
|
|
fe_EditorCanFindAgain(MWContext* context)
|
|
{
|
|
fe_FindData* find_data = CONTEXT_DATA(context)->find_data;
|
|
|
|
return ((find_data != NULL)
|
|
&&
|
|
(find_data->string != NULL)
|
|
&&
|
|
(find_data->string[0] != '\0'));
|
|
}
|
|
|
|
void
|
|
fe_EditorFindAgain(MWContext* context)
|
|
{
|
|
MWContext* top_context = XP_GetNonGridContext(context);
|
|
fe_FindData* find_data;
|
|
|
|
if (!top_context)
|
|
top_context = context;
|
|
|
|
fe_UserActivity(top_context);
|
|
|
|
find_data = CONTEXT_DATA(top_context)->find_data;
|
|
|
|
if (fe_EditorCanFindAgain(top_context))
|
|
fe_FindAndReplaceDialog(top_context, TRUE);
|
|
else
|
|
fe_Bell(context);
|
|
}
|
|
|
|
Boolean
|
|
fe_EditorCanRemoveLinks(MWContext* context)
|
|
{
|
|
ED_ElementType type = EDT_GetCurrentElementType(context);
|
|
Boolean sensitive = FALSE;
|
|
|
|
if ((type == ED_ELEMENT_TEXT ||
|
|
type == ED_ELEMENT_SELECTION ||
|
|
type == ED_ELEMENT_IMAGE)
|
|
&&
|
|
EDT_CanSetHREF(context)) {
|
|
sensitive = TRUE;
|
|
}
|
|
return sensitive;
|
|
}
|
|
|
|
void
|
|
fe_EditorInsertLinkDialogDo(MWContext* context)
|
|
{
|
|
#if 0
|
|
if (EDT_IS_NEW_DOCUMENT(context)) {
|
|
if (!FE_CheckAndSaveDocument(context))
|
|
return;
|
|
}
|
|
#endif
|
|
|
|
fe_EditorPropertiesDialogDo(context, XFE_EDITOR_PROPERTIES_LINK_INSERT);
|
|
}
|
|
|
|
/* -------------------------------------------------------------- */
|
|
|
|
/*
|
|
* Size Group.
|
|
*/
|
|
|
|
void
|
|
fe_EditorFontSizeSet(MWContext* context, ED_FontSize edt_size)
|
|
{
|
|
if (edt_size >= EDT_FONTSIZE_MIN && edt_size <= EDT_FONTSIZE_MAX) {
|
|
#if 1
|
|
EDT_SetFontSize(context, edt_size);
|
|
#else
|
|
/* this does not work */
|
|
EDT_CharacterData* edt_cdata = EDT_GetCharacterData(context);
|
|
edt_cdata->mask = TF_FONT_SIZE;
|
|
edt_cdata->values = TF_FONT_SIZE;
|
|
edt_cdata->iSize = edt_size;
|
|
EDT_SetCharacterData(context, edt_cdata);
|
|
EDT_FreeCharacterData(edt_cdata);
|
|
#endif
|
|
} else {
|
|
fe_Bell(context);
|
|
}
|
|
|
|
fe_EditorUpdateToolbar(context, TF_FONT_SIZE);
|
|
}
|
|
|
|
ED_FontSize
|
|
fe_EditorFontSizeGet(MWContext* context)
|
|
{
|
|
EDT_CharacterData* edt_cdata = EDT_GetCharacterData(context);
|
|
intn edt_size;
|
|
|
|
if (edt_cdata != NULL) {
|
|
edt_size = edt_cdata->iSize;
|
|
if (edt_size < EDT_FONTSIZE_MIN)
|
|
edt_size = ED_FONTSIZE_ZERO;
|
|
EDT_FreeCharacterData(edt_cdata);
|
|
} else {
|
|
edt_size = ED_FONTSIZE_DEFAULT;
|
|
}
|
|
|
|
return (ED_FontSize)edt_size;
|
|
}
|
|
|
|
Widget
|
|
fe_OptionMenuSetHistory(Widget menu, unsigned index)
|
|
{
|
|
Arg args[4];
|
|
Cardinal n;
|
|
Widget cascade;
|
|
Widget popup_menu;
|
|
WidgetList children;
|
|
Cardinal nchildren;
|
|
|
|
/*
|
|
* Update the label, and set the position of the popup.
|
|
*/
|
|
cascade = XmOptionButtonGadget(menu);
|
|
|
|
/*
|
|
* Get the popup menu from the cascade.
|
|
*/
|
|
n = 0;
|
|
XtSetArg(args[n], XmNsubMenuId, &popup_menu); n++;
|
|
XtGetValues(cascade, args, n);
|
|
|
|
/*
|
|
* Get the children of the popup.
|
|
*/
|
|
n = 0;
|
|
XtSetArg(args[n], XmNchildren, &children); n++;
|
|
XtSetArg(args[n], XmNnumChildren, &nchildren); n++;
|
|
XtGetValues(popup_menu, args, n);
|
|
if (index < nchildren) {
|
|
/*
|
|
* Finally, set the Nth button as history.
|
|
*/
|
|
n = 0;
|
|
XtSetArg(args[n], XmNmenuHistory, children[index]); n++;
|
|
/* NOTE: set it on the top level menu (strange) */
|
|
XtSetValues(menu, args, n);
|
|
|
|
return children[index];
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
/* -------------------------------------------------------------- */
|
|
/*
|
|
* Text Attribute set.
|
|
*/
|
|
void
|
|
fe_EditorCharacterPropertiesSet(MWContext* context, ED_TextFormat values)
|
|
{
|
|
EDT_CharacterData cdata;
|
|
|
|
memset(&cdata, 0, sizeof(EDT_CharacterData));
|
|
|
|
cdata.mask = TF_ALL_MASK;
|
|
if (values == TF_NONE) { /* pop everything */
|
|
cdata.values = 0;
|
|
} else if (values == (TF_SERVER|TF_SCRIPT)) {
|
|
cdata.mask = TF_SERVER|TF_SCRIPT;
|
|
cdata.values = 0;
|
|
} else if (values == TF_SERVER || values == TF_SCRIPT) {
|
|
cdata.mask = TF_SERVER|TF_SCRIPT;
|
|
cdata.values = values;
|
|
} else {
|
|
values &= TF_ALL_MASK; /* don't let them shoot themselves */
|
|
cdata.values = values;
|
|
}
|
|
|
|
EDT_SetCharacterData(context, &cdata);
|
|
|
|
fe_EditorUpdateToolbar(context, values);
|
|
}
|
|
|
|
ED_TextFormat
|
|
fe_EditorCharacterPropertiesGet(MWContext* context)
|
|
{
|
|
EDT_CharacterData* edt_cdata = EDT_GetCharacterData(context);
|
|
ED_TextFormat values;
|
|
|
|
if (edt_cdata != NULL) {
|
|
values = edt_cdata->values;
|
|
EDT_FreeCharacterData(edt_cdata);
|
|
} else {
|
|
values = TF_NONE;
|
|
}
|
|
|
|
return values;
|
|
}
|
|
|
|
void
|
|
fe_EditorDoPoof(MWContext* context)
|
|
{
|
|
Boolean clear_link = TRUE;
|
|
EDT_CharacterData* pData;
|
|
|
|
if (EDT_SelectionContainsLink(context)) {
|
|
/*Do you want to remove the link?*/
|
|
if (!XFE_Confirm(context,
|
|
XP_GetString(XFE_EDITOR_WARNING_REMOVE_LINK)))
|
|
clear_link = FALSE;
|
|
}
|
|
|
|
if (clear_link) {
|
|
EDT_FormatCharacter(context, TF_NONE);
|
|
} else {
|
|
pData = EDT_GetCharacterData(context);
|
|
if (pData) {
|
|
pData->mask = ~TF_HREF;
|
|
pData->values = TF_NONE;
|
|
EDT_SetCharacterData(context, pData);
|
|
EDT_FreeCharacterData(pData);
|
|
}
|
|
}
|
|
fe_EditorUpdateToolbar(context, 0);
|
|
}
|
|
|
|
/* -------------------------------------------------------------- */
|
|
/*
|
|
* Horizontal Rule.
|
|
*/
|
|
void
|
|
fe_EditorInsertHorizontalRule(MWContext* context)
|
|
{
|
|
/*
|
|
* Hrule is so simple, just use default values
|
|
* instead of bringing up properties dialog
|
|
*/
|
|
EDT_HorizRuleData *pData = EDT_NewHorizRuleData();
|
|
if (pData) {
|
|
EDT_InsertHorizRule(context, pData);
|
|
EDT_FreeHorizRuleData(pData);
|
|
}
|
|
}
|
|
|
|
void
|
|
fe_EditorIndent(MWContext* context, Boolean is_indent)
|
|
{
|
|
if (is_indent)
|
|
EDT_Outdent(context);
|
|
else
|
|
EDT_Indent(context);
|
|
}
|
|
|
|
/* -------------------------------------------------------------- */
|
|
/*
|
|
* Align Set.
|
|
*/
|
|
void
|
|
fe_EditorAlignSet(MWContext* pMWContext, ED_Alignment align)
|
|
{
|
|
ED_ElementType type;
|
|
|
|
if (EDT_IsInsertPointInTable(pMWContext))
|
|
{
|
|
EDT_SetTableAlign(pMWContext, align);
|
|
return;
|
|
}
|
|
|
|
type = EDT_GetCurrentElementType(pMWContext);
|
|
|
|
switch ( type ){
|
|
case ED_ELEMENT_HRULE:
|
|
{
|
|
EDT_HorizRuleData* pData = EDT_GetHorizRuleData(pMWContext);
|
|
if ( pData ){
|
|
pData->align = align;
|
|
EDT_SetHorizRuleData(pMWContext, pData);
|
|
}
|
|
break;
|
|
}
|
|
default: /* For Images, Text, or selection, this will do all: */
|
|
EDT_SetParagraphAlign( pMWContext, align );
|
|
break;
|
|
}
|
|
|
|
fe_EditorUpdateToolbar(pMWContext, 0);
|
|
}
|
|
|
|
ED_Alignment
|
|
fe_EditorAlignGet(MWContext* pMWContext)
|
|
{
|
|
ED_ElementType type = EDT_GetCurrentElementType(pMWContext);
|
|
EDT_HorizRuleData h_data;
|
|
|
|
if (type == ED_ELEMENT_HRULE) {
|
|
fe_EditorHorizontalRulePropertiesGet(pMWContext, &h_data);
|
|
return h_data.align;
|
|
} else { /* For Images, Text, or selection, this will do all: */
|
|
return EDT_GetParagraphAlign(pMWContext);
|
|
}
|
|
}
|
|
|
|
/* -------------------------------------------------------------- */
|
|
/*
|
|
* List Set.
|
|
*/
|
|
void
|
|
fe_EditorToggleList(MWContext* context, intn tag_type)
|
|
{
|
|
EDT_ToggleList(context, tag_type);
|
|
|
|
fe_EditorUpdateToolbar(context, 0);
|
|
}
|
|
|
|
/* -------------------------------------------------------------- */
|
|
/*
|
|
* Paragraph Styles Set.
|
|
*/
|
|
TagType
|
|
fe_EditorParagraphPropertiesGet(MWContext* context)
|
|
{
|
|
return EDT_GetParagraphFormatting(context);
|
|
}
|
|
|
|
void
|
|
fe_EditorParagraphPropertiesSet(MWContext* context, TagType type)
|
|
{
|
|
#if 1
|
|
TagType paragraph_style = EDT_GetParagraphFormatting(context);
|
|
|
|
if (type != paragraph_style) {
|
|
EDT_MorphContainer(context, type);
|
|
|
|
fe_EditorUpdateToolbar(context, 0);
|
|
}
|
|
#else
|
|
/*
|
|
* This seems like the correct code, as it would mean the toolbar
|
|
* menu and the properties dialog have the same effect when you set
|
|
* a list. But this would be different from the Windows version.
|
|
* Do above, the same as Windows.
|
|
*/
|
|
if (type == P_LIST_ITEM) {
|
|
EDT_ListData list_data;
|
|
|
|
list_data.iTagType = P_UNUM_LIST;
|
|
list_data.eType = ED_LIST_TYPE_DISC;
|
|
list_data.bCompact = FALSE;
|
|
|
|
fe_EditorParagraphPropertiesSetAll(context, type, &list_data,
|
|
ED_ALIGN_DEFAULT);
|
|
} else {
|
|
fe_EditorParagraphPropertiesSetAll(context, type, NULL,
|
|
ED_ALIGN_DEFAULT);
|
|
}
|
|
#endif
|
|
}
|
|
|
|
void
|
|
fe_EditorObjectPropertiesDialogDo(MWContext* context)
|
|
{
|
|
fe_EditorPropertiesDialogType type;
|
|
ED_ElementType e_type = EDT_GetCurrentElementType(context);
|
|
|
|
if (e_type == ED_ELEMENT_HRULE)
|
|
type = XFE_EDITOR_PROPERTIES_HRULE;
|
|
else if (e_type == ED_ELEMENT_IMAGE)
|
|
type = XFE_EDITOR_PROPERTIES_IMAGE;
|
|
else if (e_type == ED_ELEMENT_TARGET)
|
|
type = XFE_EDITOR_PROPERTIES_TARGET;
|
|
else if (e_type == ED_ELEMENT_UNKNOWN_TAG)
|
|
type = XFE_EDITOR_PROPERTIES_HTML_TAG;
|
|
else { /* character */
|
|
if (EDT_GetHREF(context))
|
|
type = XFE_EDITOR_PROPERTIES_LINK;
|
|
else
|
|
type = XFE_EDITOR_PROPERTIES_CHARACTER;
|
|
}
|
|
|
|
/*
|
|
* Validate that we can do this kind of dialog right now.
|
|
*/
|
|
if (fe_EditorPropertiesDialogCanDo(context, type)) {
|
|
fe_EditorPropertiesDialogDo(context, type);
|
|
} else {
|
|
fe_Bell(context);
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Caret handling stuff. The caret is just a timed draw onto the
|
|
* screen, it doesn't exist in the image.
|
|
*/
|
|
#define FE_CARET_DEFAULT_TIME 500
|
|
#define FE_CARET_FLAGS_BLANK 0x1
|
|
#define FE_CARET_FLAGS_DRAW 0x2
|
|
#define FE_CARET_FLAGS_XOR 0x4
|
|
#define FE_CARET_DEFAULT_WIDTH 5
|
|
|
|
static void
|
|
fe_caret_draw(MWContext *context)
|
|
{
|
|
fe_EditorCaretData* data = &EDITOR_CONTEXT_DATA(context)->caret_data;
|
|
Widget drawing_area = CONTEXT_DATA(context)->drawing_area;
|
|
Display* dpy = XtDisplay(drawing_area);
|
|
Window win = XtWindow(drawing_area);
|
|
GC gc;
|
|
XGCValues gcv;
|
|
int x = data->x;
|
|
int y = data->y;
|
|
unsigned width = data->width;
|
|
unsigned height = data->height;
|
|
|
|
#ifdef DEBUG_rhess2
|
|
fprintf(stderr, "fe_caret_draw::[ ]\n");
|
|
#endif
|
|
|
|
memset(&gcv, ~0, sizeof(gcv));
|
|
#if 0
|
|
{
|
|
LO_Color text_color;
|
|
LO_Color bg_color;
|
|
fe_EditorDocumentGetColors(context, NULL, &bg_color, &text_color,
|
|
NULL, NULL, NULL);
|
|
gcv.foreground = fe_GetPixel(context,
|
|
text_color.red,
|
|
text_color.green,
|
|
text_color.blue);
|
|
}
|
|
#else
|
|
gcv.foreground = CONTEXT_DATA(context)->fg_pixel;
|
|
#endif
|
|
gc = fe_GetGC(drawing_area, GCForeground, &gcv);
|
|
|
|
x -= CONTEXT_DATA(context)->document_x;
|
|
y -= CONTEXT_DATA(context)->document_y;
|
|
|
|
if ((width & 0x1) != 1)
|
|
width++;
|
|
|
|
/*
|
|
* Hack, hack, hack. Do something pretty david!
|
|
*/
|
|
#if 1
|
|
XDrawLine(dpy, win, gc, x, y, x + width - 1, y);
|
|
XDrawLine(dpy, win, gc, x + (width/2), y, x + (width/2), y + height - 1);
|
|
XDrawLine(dpy, win, gc, x, y + height - 1, x + width - 1, y + height - 1);
|
|
#else
|
|
XDrawRectangle(dpy, win, gc, x, y, width-1, height-1);
|
|
#endif
|
|
|
|
XFlush(dpy);
|
|
}
|
|
|
|
|
|
static void
|
|
fe_caret_undraw(MWContext *context)
|
|
{
|
|
fe_EditorCaretData* data = &EDITOR_CONTEXT_DATA(context)->caret_data;
|
|
Widget drawing_area = CONTEXT_DATA (context)->drawing_area;
|
|
Display* dpy = XtDisplay(drawing_area);
|
|
#ifdef DONT_rhess
|
|
Window win = XtWindow(drawing_area);
|
|
GC gc;
|
|
XGCValues gcv;
|
|
Visual* visual;
|
|
int visual_depth;
|
|
#endif
|
|
int x = data->x;
|
|
int y = data->y;
|
|
unsigned width = data->width;
|
|
unsigned height = data->height;
|
|
|
|
#ifdef DEBUG_rhess2
|
|
fprintf(stderr, "fe_caret_undraw::[ ]\n");
|
|
#endif
|
|
{
|
|
int32 ex = x;
|
|
int32 ey = y;
|
|
int32 ew = width + 1;
|
|
int32 eh = height + 1;
|
|
|
|
if (width > 0 && height > 0) {
|
|
/*
|
|
* NOTE: don't refresh if it's a 0 pixel box...
|
|
*/
|
|
#ifdef DEBUG_rhess2
|
|
fprintf(stderr, "fe_caret_undraw::[ %d, %d ][ %d, %d ]\n",
|
|
ex, ey, ew, eh);
|
|
#endif
|
|
fe_RefreshArea(context, ex, ey, ew, eh);
|
|
|
|
XFlush(dpy);
|
|
}
|
|
}
|
|
#ifdef DONT_rhess
|
|
{
|
|
XExposeEvent event;
|
|
|
|
/* generate an expose */
|
|
event.type = Expose;
|
|
event.serial = 0;
|
|
event.send_event = True;
|
|
event.display = XtDisplay(CONTEXT_DATA(context)->drawing_area);
|
|
event.window = XtWindow(CONTEXT_DATA(context)->drawing_area);
|
|
event.x = x - CONTEXT_DATA(context)->document_x;
|
|
event.y = y - CONTEXT_DATA(context)->document_y;
|
|
event.width = width + 1;
|
|
event.height = height + 1;
|
|
event.count = 0;
|
|
XSendEvent(event.display, event.window,
|
|
False, ExposureMask, (XEvent*)&event);
|
|
}
|
|
XFlush(dpy);
|
|
#endif
|
|
}
|
|
|
|
static void
|
|
fe_caret_update(MWContext* context, XtIntervalId *id)
|
|
{
|
|
fe_EditorCaretData* data = &EDITOR_CONTEXT_DATA(context)->caret_data;
|
|
|
|
#ifdef DONT_rhess
|
|
if (data->showing) {
|
|
fe_caret_undraw(context);
|
|
data->showing = FALSE;
|
|
} else {
|
|
fe_caret_draw(context);
|
|
data->showing = TRUE;
|
|
}
|
|
|
|
if (data->running) {
|
|
data->timer_id = XtAppAddTimeOut(fe_XtAppContext,
|
|
data->time,
|
|
(XtTimerCallbackProc)fe_caret_update,
|
|
context);
|
|
}
|
|
#else
|
|
if (data->showing) {
|
|
fe_caret_draw(context);
|
|
} else {
|
|
fe_caret_undraw(context);
|
|
}
|
|
|
|
data->timer_id = 0;
|
|
#endif
|
|
}
|
|
|
|
static void
|
|
fe_caret_cancel(MWContext* context)
|
|
{
|
|
fe_EditorCaretData* data = &EDITOR_CONTEXT_DATA(context)->caret_data;
|
|
|
|
#ifdef DEBUG_rhess2
|
|
fprintf(stderr, "fe_caret_cancel::[ ]\n");
|
|
#endif
|
|
|
|
#ifdef DONT_rhess
|
|
if (data->running)
|
|
XtRemoveTimeOut(data->timer_id);
|
|
|
|
data->running = FALSE;
|
|
|
|
if (data->showing)
|
|
fe_caret_undraw(context);
|
|
data->showing = FALSE;
|
|
|
|
if (data->backing_store) {
|
|
XFreePixmap(XtDisplay(CONTEXT_DATA(context)->drawing_area),
|
|
data->backing_store);
|
|
data->backing_store = 0;
|
|
}
|
|
#else
|
|
data->running = False;
|
|
if (data->showing) {
|
|
data->showing = False;
|
|
fe_caret_update(context, NULL);
|
|
}
|
|
#endif
|
|
}
|
|
|
|
static void
|
|
fe_caret_pause(MWContext* context)
|
|
{
|
|
fe_EditorCaretData* data = &EDITOR_CONTEXT_DATA(context)->caret_data;
|
|
|
|
#ifdef DEBUG_rhess2
|
|
fprintf(stderr, "fe_caret_pause::[ ]\n");
|
|
#endif
|
|
|
|
if (!data->running) {
|
|
/* already paused or never started */
|
|
return;
|
|
}
|
|
#ifdef DONT_rhess
|
|
else {
|
|
XtRemoveTimeOut(data->timer_id);
|
|
}
|
|
|
|
data->running = FALSE;
|
|
|
|
if (data->showing == FALSE) /* always pause showing */
|
|
fe_caret_update(context, NULL);
|
|
#else
|
|
data->running = False;
|
|
#ifdef HOT_CARET_rhess
|
|
if (!data->showing) {
|
|
data->showing = True;
|
|
#else
|
|
if (data->showing) {
|
|
data->showing = False;
|
|
#endif
|
|
fe_caret_update(context, NULL);
|
|
}
|
|
#endif /* DONT_rhess */
|
|
}
|
|
|
|
static void
|
|
fe_caret_unpause(MWContext* context)
|
|
{
|
|
fe_EditorCaretData* data = &EDITOR_CONTEXT_DATA(context)->caret_data;
|
|
|
|
#ifdef DEBUG_rhess2
|
|
fprintf(stderr, "fe_caret_unpause::[ ]\n");
|
|
#endif
|
|
|
|
#ifdef DONT_rhess
|
|
if (data->running == TRUE || data->showing == FALSE) /* not paused */
|
|
return;
|
|
|
|
data->showing = FALSE;
|
|
data->running = TRUE;
|
|
|
|
/*
|
|
* Draw and set timer
|
|
*/
|
|
fe_caret_update(context, NULL);
|
|
#else
|
|
if ( data->running || data->showing ) /* not paused */
|
|
return;
|
|
|
|
data->running = True;
|
|
|
|
if (!data->showing) {
|
|
data->showing = True;
|
|
fe_caret_update(context, NULL);
|
|
}
|
|
#endif
|
|
}
|
|
|
|
static void
|
|
fe_caret_begin(MWContext* context)
|
|
{
|
|
fe_EditorCaretData* data = &EDITOR_CONTEXT_DATA(context)->caret_data;
|
|
|
|
#ifdef DEBUG_rhess2
|
|
fprintf(stderr, "fe_caret_begin::[ ]\n");
|
|
#endif
|
|
|
|
#ifdef DONT_rhess
|
|
data->showing = FALSE;
|
|
data->time = FE_CARET_DEFAULT_TIME;
|
|
data->running = TRUE;
|
|
|
|
fe_caret_update(context, NULL);
|
|
#else
|
|
data->time = 0;
|
|
data->running = True;
|
|
if (!data->showing) {
|
|
data->showing = True;
|
|
fe_caret_update(context, NULL);
|
|
}
|
|
#endif
|
|
|
|
#ifdef XFE_EDITOR_IM_SUPPORT
|
|
fe_editor_im_set_coords(context);
|
|
#endif /* XFE_EDITOR_IM_SUPPORT */
|
|
}
|
|
|
|
void
|
|
fe_EditorRefreshArea(MWContext* context, int x, int y, unsigned w, unsigned h)
|
|
{
|
|
fe_EditorCaretData* data = &EDITOR_CONTEXT_DATA(context)->caret_data;
|
|
|
|
if (
|
|
#if 0
|
|
/*
|
|
* Don't use x, as LO_RefreshArea tends to redraw full width
|
|
* anyway.
|
|
*/
|
|
((x < data->x && x + w >= data->x) ||
|
|
(x < data->x + data->width && x + w >= data->x + data->width))
|
|
&&
|
|
#endif
|
|
((y < data->y && y + h >= data->y) ||
|
|
(y < data->y + data->height && y + h >= data->y + data->height))) {
|
|
|
|
#ifdef DONT_rhess
|
|
if (data->backing_store) {
|
|
XFreePixmap(XtDisplay(CONTEXT_DATA(context)->drawing_area),
|
|
data->backing_store);
|
|
data->backing_store = 0;
|
|
}
|
|
|
|
if (data->running) {
|
|
XtRemoveTimeOut(data->timer_id);
|
|
data->showing = TRUE; /* force restart of sequence next */
|
|
}
|
|
if (data->showing) {
|
|
data->showing = FALSE;
|
|
fe_caret_update(context, NULL);
|
|
}
|
|
#else
|
|
if (data->showing) {
|
|
fe_caret_update(context, NULL);
|
|
}
|
|
#endif
|
|
}
|
|
}
|
|
|
|
static int fe_LastCoffeeScroll_X = 0;
|
|
static int fe_LastCoffeeScroll_Y = 0;
|
|
|
|
static void
|
|
fe_editor_keep_cursor_visible(MWContext* context)
|
|
{
|
|
fe_EditorCaretData* c_data = &EDITOR_CONTEXT_DATA(context)->caret_data;
|
|
int x = CONTEXT_DATA(context)->document_x;
|
|
int y = CONTEXT_DATA(context)->document_y;
|
|
unsigned win_height = CONTEXT_DATA(context)->scrolled_height;
|
|
unsigned win_width = CONTEXT_DATA(context)->scrolled_width;
|
|
Boolean coffee_scroll = FALSE;
|
|
int fudge = 20;
|
|
int pad = 10;
|
|
|
|
#ifdef DEBUG_rhess2
|
|
fprintf(stderr, "fe_editor_keep_cursor_visible::[ ]\n");
|
|
#endif
|
|
|
|
if (c_data->y + c_data->height > y + win_height - fudge) {
|
|
y = c_data->y + c_data->height - win_height + fudge; /* fudge */
|
|
if (y + win_height > CONTEXT_DATA(context)->document_height)
|
|
y = CONTEXT_DATA(context)->document_height - win_height;
|
|
coffee_scroll = TRUE;
|
|
} else if (c_data->y < y + pad) {
|
|
y = c_data->y - pad;
|
|
if (y < 0)
|
|
y = 0;
|
|
coffee_scroll = TRUE;
|
|
}
|
|
|
|
if (c_data->x + c_data->width > x + win_width - fudge) {
|
|
x = c_data->x + c_data->width - win_width + fudge;
|
|
if (x + win_width > CONTEXT_DATA(context)->document_width)
|
|
x = CONTEXT_DATA(context)->document_width - win_width;
|
|
coffee_scroll = TRUE;
|
|
} else if (c_data->x < x + pad) {
|
|
x = c_data->x - pad;
|
|
if (x < 0)
|
|
x = 0;
|
|
coffee_scroll = TRUE;
|
|
}
|
|
|
|
if (x == fe_LastCoffeeScroll_X && y == fe_LastCoffeeScroll_Y) {
|
|
coffee_scroll = False;
|
|
}
|
|
|
|
if (coffee_scroll) {
|
|
if (c_data->showing)
|
|
fe_caret_cancel(context);
|
|
#ifdef DEBUG_rhess2
|
|
fprintf(stderr, "coffee_scroll::[ %d, %d ][ %d, %d ][ %d, %d ]\n",
|
|
x, y, c_data->x, c_data->y,
|
|
CONTEXT_DATA(context)->document_x,
|
|
CONTEXT_DATA(context)->document_y );
|
|
#endif
|
|
/*
|
|
* Collect all pending exposures before we scroll.
|
|
*/
|
|
fe_SyncExposures(context);
|
|
|
|
fe_ScrollTo(context, x, y);
|
|
}
|
|
|
|
fe_LastCoffeeScroll_X = x;
|
|
fe_LastCoffeeScroll_Y = y;
|
|
}
|
|
|
|
static void
|
|
fe_editor_keep_pointer_visible_autoscroll(XtPointer, XtIntervalId*);
|
|
|
|
static void
|
|
fe_editor_keep_pointer_visible(MWContext* context, int p_x, int p_y)
|
|
{
|
|
fe_ContextData* data = CONTEXT_DATA(context);
|
|
fe_EditorAscrollData* ascroll_data
|
|
= &EDITOR_CONTEXT_DATA(context)->ascroll_data;
|
|
Boolean coffee_scroll = FALSE;
|
|
int x = data->document_x;
|
|
int y = data->document_y;
|
|
int delta = 0;
|
|
|
|
#ifdef DEBUG_rhess2
|
|
fprintf(stderr, "fe_editor_keep_pointer_visible::[ ]\n");
|
|
#endif
|
|
|
|
if (ascroll_data->timer_id)
|
|
XtRemoveTimeOut(ascroll_data->timer_id);
|
|
|
|
ascroll_data->delta_x = 0;
|
|
ascroll_data->delta_y = 0;
|
|
|
|
if (p_y < data->document_y) {
|
|
coffee_scroll = TRUE;
|
|
y = p_y;
|
|
delta = (data->document_y - p_y);
|
|
ascroll_data->y = p_y;
|
|
ascroll_data->delta_y = -delta;
|
|
}
|
|
|
|
if (p_y > data->document_y + data->scrolled_height) {
|
|
coffee_scroll = TRUE;
|
|
y = p_y - data->scrolled_height;
|
|
delta = p_y - (data->document_y + data->scrolled_height);
|
|
ascroll_data->y = p_y;
|
|
ascroll_data->delta_y = delta;
|
|
}
|
|
|
|
if (coffee_scroll) {
|
|
/*
|
|
* Collect all pending exposures before we scroll.
|
|
*/
|
|
fe_SyncExposures(context);
|
|
|
|
fe_ScrollTo(context, x, y);
|
|
|
|
ascroll_data->timer_id =
|
|
XtAppAddTimeOut(fe_XtAppContext,
|
|
(100),
|
|
fe_editor_keep_pointer_visible_autoscroll,
|
|
(XtPointer)context);
|
|
}
|
|
}
|
|
|
|
static void
|
|
fe_editor_keep_pointer_visible_autoscroll(XtPointer closure, XtIntervalId* id)
|
|
{
|
|
MWContext* context = (MWContext*)closure;
|
|
fe_EditorAscrollData* ascroll_data;
|
|
|
|
#if DEBUG_rhess2
|
|
fprintf (stderr,"fe_editor_keep_pointer_visible_autoscroll::[ ]\n");
|
|
#endif
|
|
|
|
ascroll_data = &EDITOR_CONTEXT_DATA(context)->ascroll_data;
|
|
ascroll_data->timer_id = 0; /* because we won't pop again */
|
|
|
|
ascroll_data->x += ascroll_data->delta_x;
|
|
ascroll_data->y += ascroll_data->delta_y;
|
|
|
|
EDT_ExtendSelection(context, ascroll_data->x, ascroll_data->y);
|
|
|
|
fe_editor_keep_pointer_visible(context, ascroll_data->x, ascroll_data->y);
|
|
}
|
|
|
|
static void
|
|
fe_caret_set(MWContext* context, int x, int y, unsigned w, unsigned h)
|
|
{
|
|
fe_EditorCaretData* data = &EDITOR_CONTEXT_DATA(context)->caret_data;
|
|
|
|
#ifdef DEBUG_rhess2
|
|
fprintf(stderr, "fe_caret_set::[ ]\n");
|
|
#endif
|
|
|
|
if ( x != data->x ||
|
|
y != data->y ||
|
|
w != data->width ||
|
|
h != data->height )
|
|
{
|
|
if (data->showing)
|
|
fe_caret_cancel(context);
|
|
}
|
|
|
|
#ifdef DONT_rhess
|
|
if (data->running)
|
|
XtRemoveTimeOut(data->timer_id);
|
|
#endif
|
|
|
|
data->x = x;
|
|
data->y = y;
|
|
data->width = w;
|
|
data->height = h;
|
|
|
|
fe_editor_keep_cursor_visible(context);
|
|
}
|
|
|
|
Boolean
|
|
FE_GetCaretPosition(
|
|
MWContext *context,
|
|
LO_Position* where,
|
|
int32* caretX, int32* caretYLow, int32* caretYHigh
|
|
) {
|
|
int32 xVal;
|
|
int32 yVal;
|
|
int32 yValHigh;
|
|
|
|
#if DEBUG_rhess2
|
|
fprintf(real_stderr, "FE_GetCaretPosition::[ ]\n");
|
|
#endif
|
|
|
|
if (!context || !where->element)
|
|
return FALSE;
|
|
|
|
xVal = where->element->lo_any.x + where->element->lo_any.x_offset;
|
|
yVal = where->element->lo_any.y;
|
|
yValHigh = yVal + where->element->lo_any.height;
|
|
|
|
switch (where->element->type) {
|
|
case LO_TEXT: {
|
|
LO_TextStruct* text_data = & where->element->lo_text;
|
|
LO_TextInfo text_info;
|
|
int len_save = text_data->text_len;
|
|
|
|
if (!text_data->text_attr)
|
|
return FALSE;
|
|
|
|
if (where->position <= text_data->text_len) {
|
|
text_data->text_len = where->position;
|
|
}
|
|
XFE_GetTextInfo(context, text_data, &text_info);
|
|
text_data->text_len = len_save;
|
|
|
|
xVal += text_info.max_width - 1;
|
|
} break;
|
|
|
|
case LO_IMAGE: {
|
|
LO_ImageStruct *pLoImage = & where->element->lo_image;
|
|
if (where->position == 0) {
|
|
xVal -= 1;
|
|
} else {
|
|
xVal += pLoImage->width + 2 * pLoImage->border_width;
|
|
}
|
|
} break;
|
|
|
|
default: {
|
|
LO_Any *any = &where->element->lo_any;
|
|
if (where->position == 0) {
|
|
xVal -= 1;
|
|
} else {
|
|
xVal += any->width;
|
|
}
|
|
}
|
|
}
|
|
|
|
*caretX = xVal;
|
|
*caretYLow = yVal;
|
|
*caretYHigh = yValHigh;
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
/*
|
|
* char_offset is in characters!!!
|
|
*/
|
|
PUBLIC void
|
|
FE_DisplayTextCaret(MWContext* context, int iLocation, LO_TextStruct* text,
|
|
int char_offset)
|
|
{
|
|
int x;
|
|
int y;
|
|
unsigned width;
|
|
unsigned height;
|
|
LO_TextInfo text_info;
|
|
int16 save_len;
|
|
|
|
#if DEBUG_rhess2
|
|
fprintf (stderr,"FE_DisplayTextCaret::[ ]\n");
|
|
#endif
|
|
|
|
/*
|
|
* Get info extent info about the first <char_offset> characters of
|
|
* text.
|
|
*/
|
|
save_len = text->text_len;
|
|
text->text_len = char_offset;
|
|
XFE_GetTextInfo(context, text, &text_info);
|
|
x = text->x + text->x_offset + text_info.max_width;
|
|
y = text->y + text->y_offset;
|
|
height = text_info.ascent + text_info.descent;
|
|
|
|
text->text_len = save_len;
|
|
|
|
width = FE_CARET_DEFAULT_WIDTH;
|
|
x -= (FE_CARET_DEFAULT_WIDTH/2) + 1; /* middle of char and back */
|
|
|
|
fe_caret_set(context, x, y, width, height);
|
|
fe_caret_begin(context);
|
|
}
|
|
|
|
void
|
|
FE_DisplayImageCaret(
|
|
MWContext* context,
|
|
LO_ImageStruct* image,
|
|
ED_CaretObjectPosition caretPos)
|
|
{
|
|
int x;
|
|
int y;
|
|
unsigned width;
|
|
unsigned height;
|
|
|
|
#if DEBUG_rhess2
|
|
fprintf (stderr,"FE_DisplayImageCaret::[ ]\n");
|
|
#endif
|
|
|
|
/*
|
|
* Get info extent info about the first <char_offset> characters of
|
|
* text.
|
|
*/
|
|
x = image->x + image->x_offset;
|
|
y = image->y + image->y_offset;
|
|
width = FE_CARET_DEFAULT_WIDTH;
|
|
height = image->height + (2 * image->border_width);
|
|
|
|
if (caretPos == ED_CARET_BEFORE) {
|
|
x -= 1;
|
|
} else if (caretPos == ED_CARET_AFTER) {
|
|
x += image->width + (2 * image->border_width);
|
|
} else {
|
|
width = image->width + (2 * image->border_width);
|
|
}
|
|
|
|
fe_caret_set(context, x, y, width, height);
|
|
fe_caret_begin(context);
|
|
}
|
|
|
|
void FE_DisplayGenericCaret(MWContext * context, LO_Any * image,
|
|
ED_CaretObjectPosition caretPos )
|
|
{
|
|
int x;
|
|
int y;
|
|
unsigned width;
|
|
unsigned height;
|
|
|
|
#if DEBUG_rhess2
|
|
fprintf (stderr,"FE_DisplayGenericCaret::[ ]\n");
|
|
#endif
|
|
|
|
/*
|
|
* Get info extent info about the first <char_offset> characters of
|
|
* text.
|
|
*/
|
|
x = image->x + image->x_offset;
|
|
y = image->y + image->y_offset;
|
|
width = FE_CARET_DEFAULT_WIDTH;
|
|
height = image->line_height;
|
|
|
|
if (caretPos == ED_CARET_BEFORE) {
|
|
x -= 1;
|
|
} else if (caretPos == ED_CARET_AFTER) {
|
|
x += image->width;
|
|
} else {
|
|
width = image->width;
|
|
}
|
|
|
|
fe_caret_set(context, x, y, width, height);
|
|
fe_caret_begin(context);
|
|
}
|
|
|
|
PUBLIC void
|
|
FE_DestroyCaret(MWContext * context)
|
|
{
|
|
fe_EditorCaretData* data = &EDITOR_CONTEXT_DATA(context)->caret_data;
|
|
|
|
#if DEBUG_rhess2
|
|
fprintf(stderr, "FE_DestroyCaret::[ ]\n");
|
|
#endif
|
|
|
|
if (data->showing)
|
|
fe_caret_cancel(context);
|
|
|
|
data->time = 0;
|
|
data->timer_id = 0;
|
|
data->showing = False;
|
|
data->running = False;
|
|
data->backing_store = 0;
|
|
}
|
|
|
|
PUBLIC void
|
|
FE_ShowCaret(MWContext * context)
|
|
{
|
|
#if DEBUG_rhess2
|
|
fprintf(stderr, "FE_ShowCaret::[ ]\n");
|
|
#endif
|
|
fe_caret_begin(context);
|
|
}
|
|
|
|
/*
|
|
* Note all x, y co-ordinates are document relative.
|
|
*/
|
|
PUBLIC void
|
|
FE_DocumentChanged(MWContext* context, int32 p_y, int32 p_height)
|
|
{
|
|
int32 redraw_x = 0;
|
|
int32 redraw_y;
|
|
uint32 redraw_width = CONTEXT_DATA(context)->document_width;
|
|
uint32 redraw_height;
|
|
int32 win_y;
|
|
uint32 win_height;
|
|
int32 win_x;
|
|
int32 margin_height;
|
|
int32 margin_width;
|
|
|
|
win_y = CONTEXT_DATA(context)->document_y;
|
|
win_height = CONTEXT_DATA(context)->scrolled_height;
|
|
|
|
win_x = CONTEXT_DATA(context)->document_x;
|
|
|
|
/*
|
|
* Ok, it seems like layout doesn't take into account margins.
|
|
* If there was stuff in the margin before (because we *were*
|
|
* displaying something big, which was scrolled up), but now we
|
|
* are not, we must clear the margin. This code attempts
|
|
* to detect when we are close to the margin, and say, clear
|
|
* the whole thing. This seems kinda bogus, the back-end
|
|
* should know about margins (no?), but... djw
|
|
*/
|
|
fe_GetMargin(context, &margin_width, &margin_height);
|
|
|
|
if (p_y <= margin_height) {
|
|
p_y = 0;
|
|
if (p_height != -1)
|
|
p_height += margin_height;
|
|
}
|
|
|
|
redraw_y = p_y;
|
|
|
|
if (p_height < 0) {
|
|
redraw_height = CONTEXT_DATA(context)->document_height;
|
|
/* make sure full repaint goes to end of window at least. */
|
|
if (win_height > redraw_height)
|
|
redraw_height = win_height;
|
|
} else {
|
|
redraw_height = p_height;
|
|
}
|
|
|
|
/*
|
|
* Is doc change area above or below displayed region?
|
|
*/
|
|
if (redraw_y > win_y + win_height || redraw_y + redraw_height < win_y)
|
|
return; /* nothing to do */
|
|
|
|
/*
|
|
* Clip redraw area.
|
|
*/
|
|
if (redraw_y < win_y) {
|
|
redraw_height -= (win_y - redraw_y);
|
|
redraw_y = win_y;
|
|
}
|
|
|
|
if (redraw_y + redraw_height > win_y + win_height) {
|
|
redraw_height = win_y + win_height - redraw_y;
|
|
}
|
|
|
|
/*FIXME*/
|
|
/*
|
|
* Probably need to call CL_RefreshRegion()/Rect()
|
|
*/
|
|
#ifdef DONT_rhess
|
|
fe_ClearArea(context, redraw_x - win_x, redraw_y - win_y,
|
|
redraw_width, redraw_height);
|
|
#endif
|
|
fe_caret_cancel(context);
|
|
|
|
{
|
|
int32 ex = redraw_x;
|
|
int32 ey = redraw_y;
|
|
int32 ew = redraw_width;
|
|
int32 eh = redraw_height;
|
|
|
|
#ifdef DEBUG_rhess2
|
|
fprintf(stderr, "FE_DocumentChanged::[ %d, %d ][ %d, %d ]\n",
|
|
ex, ey, ew, eh);
|
|
#endif
|
|
fe_RefreshArea(context, ex, ey, ew, eh);
|
|
}
|
|
#ifdef DONT_rhess
|
|
{
|
|
XExposeEvent event;
|
|
|
|
/* generate an expose */
|
|
event.type = Expose;
|
|
event.serial = 0;
|
|
event.send_event = True;
|
|
event.display = XtDisplay(CONTEXT_DATA(context)->drawing_area);
|
|
event.window = XtWindow(CONTEXT_DATA(context)->drawing_area);
|
|
event.x = redraw_x - win_x;
|
|
event.y = redraw_y - win_y;
|
|
event.width = redraw_width;
|
|
event.height = redraw_height;
|
|
event.count = 0;
|
|
XSendEvent(event.display, event.window, False, ExposureMask, (XEvent*)&event);
|
|
}
|
|
#endif
|
|
}
|
|
|
|
void FE_GetDocAndWindowPosition(MWContext * context, int32 *pX, int32 *pY,
|
|
int32 *pWidth, int32 *pHeight )
|
|
{
|
|
*pX = CONTEXT_DATA (context)->document_x;
|
|
*pY = CONTEXT_DATA (context)->document_y;
|
|
*pWidth = CONTEXT_DATA (context)->scrolled_width;
|
|
*pHeight = CONTEXT_DATA (context)->scrolled_height;
|
|
}
|
|
|
|
void
|
|
FE_SetNewDocumentProperties(MWContext* context)
|
|
{
|
|
|
|
LO_Color bg_color;
|
|
LO_Color normal_color;
|
|
LO_Color link_color;
|
|
LO_Color active_color;
|
|
LO_Color followed_color;
|
|
char background_image[MAXPATHLEN];
|
|
XP_Bool use_custom;
|
|
Boolean keep_images;
|
|
|
|
/*
|
|
* Set keep images state from prefs.
|
|
*/
|
|
fe_EditorPreferencesGetLinksAndImages(context, NULL, &keep_images);
|
|
fe_EditorDocumentSetImagesWithDocument(context, keep_images);
|
|
|
|
/*
|
|
* Get editor defaults.
|
|
*/
|
|
use_custom = fe_EditorPreferencesGetColors(context,
|
|
background_image,
|
|
&bg_color,
|
|
&normal_color,
|
|
&link_color,
|
|
&active_color,
|
|
&followed_color);
|
|
|
|
/*
|
|
* Apply.
|
|
*/
|
|
if (use_custom) {
|
|
fe_EditorDocumentSetColors(context,
|
|
background_image,
|
|
FALSE,
|
|
&bg_color,
|
|
&normal_color,
|
|
&link_color,
|
|
&active_color,
|
|
&followed_color);
|
|
} else {
|
|
fe_EditorDocumentSetColors(context,
|
|
background_image, /*independent of colors*/
|
|
FALSE,
|
|
NULL,
|
|
NULL,
|
|
NULL,
|
|
NULL,
|
|
NULL);
|
|
}
|
|
|
|
/*
|
|
* Now do other stuff.
|
|
*/
|
|
/* Title */
|
|
fe_EditorDocumentSetTitle(context, NULL); /* will show Untitled */
|
|
|
|
/* Add fixed MetaData items for author, others?? */
|
|
fe_EditorDocumentSetMetaData(context,
|
|
"Author",
|
|
fe_EditorPreferencesGetAuthor(context));
|
|
fe_EditorDocumentSetMetaData(context,
|
|
"GENERATOR",
|
|
"Mozilla/X");
|
|
}
|
|
|
|
void
|
|
fe_EditorNotifyBackgroundChanged(MWContext* context)
|
|
{
|
|
#ifdef DONT_rhess
|
|
fe_EditorCaretData* data = &EDITOR_CONTEXT_DATA(context)->caret_data;
|
|
|
|
if (data->backing_store) {
|
|
XFreePixmap(XtDisplay(CONTEXT_DATA(context)->drawing_area),
|
|
data->backing_store);
|
|
data->backing_store = 0;
|
|
}
|
|
#endif
|
|
}
|
|
|
|
char*
|
|
FE_URLToLocalName(char *name)
|
|
{
|
|
char* qm;
|
|
char* hm;
|
|
char* begin;
|
|
char* end;
|
|
char* rv;
|
|
char* nameSave;
|
|
int len;
|
|
int nameLen;
|
|
|
|
/*
|
|
* Save a copy of the name as we might drop characters after '?' or '#'
|
|
*/
|
|
|
|
nameLen = strlen(name);
|
|
nameSave = XP_ALLOC(nameLen + 1);
|
|
memcpy(nameSave, name, nameLen);
|
|
nameSave[nameLen] = '\0';
|
|
|
|
/*
|
|
* Drop everything after a '?' or '#' character.
|
|
*/
|
|
qm = strchr(nameSave, '?');
|
|
hm = strchr(nameSave, '#');
|
|
|
|
if (qm && hm) {
|
|
end = (qm > hm)? qm: hm;
|
|
} else if (qm) {
|
|
end = qm;
|
|
} else if (hm) {
|
|
end = hm;
|
|
} else {
|
|
end = nameSave + nameLen;
|
|
}
|
|
|
|
*end = '\0';
|
|
|
|
/*
|
|
* Then get the basename of what's left.
|
|
*/
|
|
if ((begin = strrchr(nameSave, '/')) != NULL && begin[1] != '\0') {
|
|
begin++;
|
|
} else {
|
|
begin = nameSave;
|
|
}
|
|
|
|
len = end - begin;
|
|
rv = malloc(len + 1);
|
|
memcpy(rv, begin, len);
|
|
rv[len] = '\0';
|
|
|
|
XP_FREE(nameSave);
|
|
return rv;
|
|
}
|
|
|
|
Boolean FE_EditorPrefConvertFileCaseOnWrite(void)
|
|
{
|
|
fprintf (real_stderr,"FE_EditorPrefConvertFileCaseOnWrite\n");
|
|
return TRUE;
|
|
}
|
|
|
|
void
|
|
FE_FinishedSave(MWContext* context, int status,
|
|
char *pDestURL, int iFileNumber)
|
|
{
|
|
}
|
|
|
|
#if 0
|
|
static void
|
|
fe_editor_copyright_hint_dialog(MWContext* context)
|
|
{
|
|
if (!fe_globalPrefs.editor_copyright_hint)
|
|
return;
|
|
|
|
if (fe_HintDialog(context, XP_GetString(XFE_ERROR_COPYRIGHT_HINT))) {
|
|
fe_globalPrefs.editor_copyright_hint = FALSE;
|
|
|
|
/*
|
|
* Save options.
|
|
*/
|
|
if (!XFE_SavePrefs((char *)fe_globalData.user_prefs_file,
|
|
&fe_globalPrefs)) {
|
|
fe_perror(context, XP_GetString(XFE_ERROR_SAVING_OPTIONS));
|
|
}
|
|
}
|
|
}
|
|
#endif
|
|
|
|
/*
|
|
* Crock timeout callback that is used by fe_EditorGetUrlExitRoutine() to
|
|
* kill a failed Frame (you opened a non-html doc say).
|
|
*/
|
|
static void
|
|
fe_editor_exit_timeout(XtPointer closure, XtIntervalId* id)
|
|
{
|
|
MWContext *context = (MWContext *)closure;
|
|
|
|
fe_EditorKill(context);
|
|
}
|
|
|
|
void
|
|
fe_EditorGetUrlExitRoutine(MWContext* context, URL_Struct* url, int status)
|
|
{
|
|
if (status < 0) {
|
|
/*
|
|
* Make sure that it's not a publish. This code is only
|
|
* meant for fe_GetURL() calls. Make sure the user gets to see
|
|
* the error dialog, then request a delete on the editor.
|
|
*/
|
|
if (url != NULL && url->files_to_post == NULL) { /* not publish */
|
|
|
|
/*
|
|
* Wait for popup menus to go away.
|
|
*/
|
|
/* we need to check context->type because under certain cases we
|
|
* reach this loop after the editor has been killed so context is
|
|
* garbage and if we don't check then ContextHasPopups fail
|
|
*/
|
|
while (context && context->type == MWContextEditor &&
|
|
fe_ContextHasPopups(context))
|
|
fe_EventLoop();
|
|
|
|
/*
|
|
* Then request a dissapearing act.
|
|
*/
|
|
#ifdef ENDER
|
|
if (! EDITOR_CONTEXT_DATA(context)->embedded)
|
|
#endif /* ENDER */
|
|
XtAppAddTimeOut(fe_XtAppContext, 0, fe_editor_exit_timeout,
|
|
(XtPointer)context);
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
/*
|
|
* Editor calls us when we are finished loading
|
|
* We force saving document if not editing local file
|
|
*/
|
|
void
|
|
FE_EditorDocumentLoaded(MWContext* context)
|
|
{
|
|
History_entry* hist_ent;
|
|
Bool do_save_dialog;
|
|
unsigned as_time;
|
|
Boolean as_enable;
|
|
|
|
if (!context) {
|
|
return;
|
|
}
|
|
|
|
#ifdef DEBUG_rhess2
|
|
fprintf(stderr, "editor::[ FE_EditorDocumentLoaded ]\n");
|
|
#endif
|
|
|
|
/*
|
|
* Windows does a lot more gymnastics here to determine which
|
|
* docs should or should not be saved (especially new ones).
|
|
* Please look into this and match up.
|
|
*/
|
|
/*FIXME*/
|
|
do_save_dialog = FALSE;
|
|
if ((hist_ent = SHIST_GetCurrent(&context->hist)) && hist_ent->address) {
|
|
do_save_dialog = !NET_IsLocalFileURL(hist_ent->address);
|
|
}
|
|
|
|
#if 0
|
|
/*
|
|
* Seems we don't do this anymore.
|
|
*/
|
|
if (!EDT_IS_NEW_DOCUMENT(context) && do_save_dialog) {
|
|
|
|
/*
|
|
* Tongue in cheek....
|
|
*/
|
|
fe_editor_copyright_hint_dialog(context);
|
|
}
|
|
#endif
|
|
|
|
#ifdef MOZ_MAIL_NEWS
|
|
if (context->type == MWContextMessageComposition) {
|
|
fe_MailComposeDocumentLoaded(context);
|
|
}
|
|
else
|
|
#endif
|
|
xfe2_EditorInit(context);
|
|
|
|
/*
|
|
* Set autosave period.
|
|
*/
|
|
#ifdef ENDER
|
|
if (! EDITOR_CONTEXT_DATA(context)->embedded)
|
|
#endif /* ENDER */
|
|
{
|
|
fe_EditorPreferencesGetAutoSave(context, &as_enable, &as_time);
|
|
EDT_SetAutoSavePeriod(context, as_time);
|
|
}
|
|
|
|
fe_HackEditorNotifyToolbarUpdated(context);
|
|
}
|
|
|
|
static char fe_SaveDialogSaveName[] = "saveMessageDialog";
|
|
static char fe_SaveDialogUploadName[] = "uploadMessageDialog";
|
|
static char fe_SaveDialogPrepareName[] = "prepareMessageDialog";
|
|
static char fe_SaveDialogImageLoadName[] = "imageLoadMessageDialog";
|
|
|
|
typedef enum fe_SaveDialogType
|
|
{
|
|
XFE_SAVE_DIALOG_SAVE,
|
|
XFE_SAVE_DIALOG_UPLOAD,
|
|
XFE_SAVE_DIALOG_PREPARE,
|
|
XFE_SAVE_DIALOG_IMAGELOAD
|
|
} fe_SaveDialogType;
|
|
|
|
typedef struct fe_SaveDialogInfo
|
|
{
|
|
unsigned nfiles;
|
|
unsigned current;
|
|
fe_SaveDialogType type;
|
|
} fe_SaveDialogInfo;
|
|
|
|
static void
|
|
fe_save_dialog_info_init(Widget msg_box, fe_SaveDialogType type, unsigned n)
|
|
{
|
|
fe_SaveDialogInfo* info = (fe_SaveDialogInfo*)XtNew(fe_SaveDialogInfo);
|
|
|
|
if (info == NULL)
|
|
return;
|
|
|
|
info->nfiles = n;
|
|
info->type = type;
|
|
info->current = 0;
|
|
|
|
XtVaSetValues(msg_box, XmNuserData, (XtPointer)info, 0);
|
|
}
|
|
|
|
#define XFE_SAVE_DIALOG_NEXT (-1)
|
|
|
|
static void
|
|
fe_save_dialog_info_set(Widget msg_box, char* filename, int index)
|
|
{
|
|
XtPointer user_data;
|
|
fe_SaveDialogInfo* info;
|
|
char buf[MAXPATHLEN];
|
|
int id;
|
|
char* string;
|
|
XmString xm_string;
|
|
|
|
XtVaGetValues(msg_box, XmNuserData, &user_data, 0);
|
|
if (user_data != 0) {
|
|
info = (fe_SaveDialogInfo*)user_data;
|
|
|
|
if (index == XFE_SAVE_DIALOG_NEXT)
|
|
info->current++;
|
|
else
|
|
info->current = index;
|
|
|
|
if (info->current > info->nfiles)
|
|
info->nfiles = info->current;
|
|
|
|
switch (info->type) {
|
|
case XFE_SAVE_DIALOG_IMAGELOAD:
|
|
id = XFE_LOADING_IMAGE_FILE;
|
|
break;
|
|
case XFE_SAVE_DIALOG_UPLOAD:
|
|
id = XFE_UPLOADING_FILE;
|
|
break;
|
|
case XFE_SAVE_DIALOG_PREPARE:
|
|
id = XFE_PREPARE_UPLOAD;
|
|
break;
|
|
default:
|
|
id = XFE_SAVING_FILE;
|
|
break;
|
|
}
|
|
|
|
string = XP_GetString(id);
|
|
|
|
if (!filename)
|
|
filename = "";
|
|
sprintf(buf, string, filename);
|
|
|
|
if (info->nfiles > 1) {
|
|
strcat(buf, "\n");
|
|
string = XP_GetString(XFE_FILE_N_OF_N);
|
|
sprintf(&buf[strlen(buf)], string, info->current, info->nfiles);
|
|
}
|
|
|
|
xm_string = XmStringCreateLocalized(buf);
|
|
} else {
|
|
xm_string = XmStringCreateLocalized(filename);
|
|
}
|
|
XtVaSetValues(msg_box, XmNmessageString, xm_string, 0);
|
|
XmStringFree(xm_string);
|
|
}
|
|
|
|
static fe_SaveDialogType
|
|
fe_SaveDialogGetType(Widget msg_box)
|
|
{
|
|
char* name = XtName(msg_box);
|
|
|
|
if (strcmp(name, fe_SaveDialogSaveName) == 0)
|
|
return XFE_SAVE_DIALOG_SAVE;
|
|
else if (strcmp(name, fe_SaveDialogUploadName) == 0)
|
|
return XFE_SAVE_DIALOG_UPLOAD;
|
|
else if (strcmp(name, fe_SaveDialogPrepareName) == 0)
|
|
return XFE_SAVE_DIALOG_PREPARE;
|
|
else
|
|
return XFE_SAVE_DIALOG_IMAGELOAD;
|
|
}
|
|
|
|
static void
|
|
fe_save_dialog_cancel_cb(Widget msg_box, XtPointer closure, XtPointer cb_data)
|
|
{
|
|
MWContext* context = (MWContext*)closure;
|
|
|
|
if (fe_SaveDialogGetType(msg_box) == XFE_SAVE_DIALOG_SAVE) {
|
|
EDT_SaveCancel(context);
|
|
} else {
|
|
NET_InterruptWindow(context);
|
|
}
|
|
}
|
|
|
|
static void
|
|
fe_save_dialog_destroy_cb(Widget msg_box, XtPointer closure, XtPointer cb_data)
|
|
{
|
|
MWContext* context = (MWContext*)closure;
|
|
XtPointer user_data;
|
|
|
|
if (CONTEXT_DATA(context)->posted_msg_box == msg_box)
|
|
CONTEXT_DATA(context)->posted_msg_box = NULL;
|
|
|
|
XtVaGetValues(msg_box, XmNuserData, &user_data, 0);
|
|
if (user_data != NULL)
|
|
XtFree(user_data);
|
|
}
|
|
|
|
static Widget
|
|
fe_save_dialog_create(MWContext* context, char* name)
|
|
{
|
|
Widget mainw = CONTEXT_WIDGET(context);
|
|
Widget msg_box;
|
|
Visual *v = 0;
|
|
Colormap cmap = 0;
|
|
Cardinal depth = 0;
|
|
Arg args [20];
|
|
Cardinal n;
|
|
|
|
if (CONTEXT_DATA(context)->posted_msg_box != NULL)
|
|
return CONTEXT_DATA(context)->posted_msg_box;
|
|
|
|
XtVaGetValues(mainw, XtNvisual, &v, XtNcolormap, &cmap,
|
|
XtNdepth, &depth, 0);
|
|
|
|
n = 0;
|
|
XtSetArg(args[n], XmNvisual, v); n++;
|
|
XtSetArg(args[n], XmNdepth, depth); n++;
|
|
XtSetArg(args[n], XmNcolormap, cmap); n++;
|
|
XtSetArg(args[n], XmNallowShellResize, TRUE); n++;
|
|
XtSetArg(args[n], XmNtransientFor, mainw); n++;
|
|
XtSetArg(args[n], XmNdialogStyle, XmDIALOG_FULL_APPLICATION_MODAL); n++;
|
|
XtSetArg(args[n], XmNdialogType, XmDIALOG_MESSAGE); n++;
|
|
XtSetArg(args[n], XmNdeleteResponse, XmDO_NOTHING); n++;
|
|
XtSetArg(args[n], XmNautoUnmanage, FALSE); n++;
|
|
XtSetArg(args[n], XmNresizePolicy, XmRESIZE_GROW); n++;
|
|
|
|
msg_box = XmCreateMessageDialog(mainw, name, args, n);
|
|
|
|
fe_UnmanageChild_safe(XmMessageBoxGetChild(msg_box, XmDIALOG_OK_BUTTON));
|
|
fe_UnmanageChild_safe(XmMessageBoxGetChild(msg_box, XmDIALOG_HELP_BUTTON));
|
|
|
|
XtAddCallback(msg_box, XmNcancelCallback, fe_save_dialog_cancel_cb,
|
|
(XtPointer)context);
|
|
XtAddCallback(msg_box, XmNdestroyCallback, fe_save_dialog_destroy_cb,
|
|
(XtPointer)context);
|
|
|
|
CONTEXT_DATA(context)->posted_msg_box = msg_box;
|
|
|
|
return msg_box;
|
|
}
|
|
|
|
static void
|
|
fe_destroy_parent_cb(Widget widget, XtPointer closure, XtPointer cbd)
|
|
{
|
|
XtDestroyWidget(XtParent(widget));
|
|
}
|
|
|
|
static void
|
|
fe_save_dialog_destroy(Widget msg_box)
|
|
{
|
|
Widget shell = XtParent(msg_box);
|
|
Cardinal num_popups = XfeNumPopups(shell);
|
|
Widget* popups = XfePopupList(shell);
|
|
|
|
/*
|
|
* If we have a stacked popup on us, them wait for that
|
|
* popup to destroy before we destroy ourselves. Do this by
|
|
* setting a destroy callback on the popup that destroys
|
|
* it's parent (us). Amongst other things we need to do this
|
|
* because server side error messages get stacked on the save
|
|
* progress dialog.
|
|
*/
|
|
if (num_popups > 0) { /* anyone who adds to me after this is a loser */
|
|
XtAddCallback(popups[0], XmNdestroyCallback, fe_destroy_parent_cb, 0);
|
|
} else {
|
|
XtDestroyWidget(msg_box);
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Dialog to give feedback and allow canceling, overwrite protection
|
|
* when downloading remote files
|
|
*
|
|
* put up a save dialog that says "i'm saving foo right now", cancel?
|
|
* and return right away.
|
|
*/
|
|
void
|
|
FE_SaveDialogCreate(MWContext* context, int nfiles, ED_SaveDialogType saveType)
|
|
{
|
|
Widget msg_box;
|
|
char* name;
|
|
fe_SaveDialogType type;
|
|
|
|
if (CONTEXT_DATA(context)->posted_msg_box != NULL)
|
|
return;
|
|
|
|
switch (saveType) {
|
|
case ED_SAVE_DLG_PUBLISH:
|
|
name = fe_SaveDialogUploadName;
|
|
type = XFE_SAVE_DIALOG_UPLOAD;
|
|
break;
|
|
case ED_SAVE_DLG_PREPARE_PUBLISH:
|
|
name = fe_SaveDialogPrepareName;
|
|
type = XFE_SAVE_DIALOG_PREPARE;
|
|
break;
|
|
default:
|
|
name = fe_SaveDialogSaveName;
|
|
type = XFE_SAVE_DIALOG_SAVE;
|
|
break;
|
|
}
|
|
|
|
msg_box = fe_save_dialog_create(context, name);
|
|
|
|
fe_save_dialog_info_init(msg_box, type, nfiles);
|
|
|
|
CONTEXT_DATA(context)->posted_msg_box = msg_box;
|
|
|
|
/* this is here so that we know when the file saying is over */
|
|
CONTEXT_DATA(context)->is_file_saving = True;
|
|
|
|
/*
|
|
* Postpone until we have something to say..
|
|
XtManageChild(msg_box);
|
|
*/
|
|
}
|
|
|
|
void
|
|
FE_SaveDialogSetFilename(MWContext* context, char* filename)
|
|
{
|
|
Widget msg_box = CONTEXT_DATA(context)->posted_msg_box;
|
|
|
|
if (!msg_box)
|
|
return;
|
|
|
|
fe_save_dialog_info_set(msg_box, filename, XFE_SAVE_DIALOG_NEXT);
|
|
|
|
if (!XtIsManaged(msg_box))
|
|
XtManageChild(msg_box);
|
|
|
|
XfeUpdateDisplay(msg_box);
|
|
}
|
|
|
|
void
|
|
FE_SaveDialogDestroy(MWContext* context, int status, char* file_url)
|
|
{
|
|
Widget msg_box = CONTEXT_DATA(context)->posted_msg_box;
|
|
fe_SaveDialogType type;
|
|
|
|
if (msg_box == NULL)
|
|
return;
|
|
|
|
type = fe_SaveDialogGetType(msg_box);
|
|
|
|
if (type == XFE_SAVE_DIALOG_SAVE) {
|
|
CONTEXT_DATA(context)->save_file_url = file_url;
|
|
CONTEXT_DATA(context)->file_save_status = status;
|
|
}
|
|
|
|
fe_save_dialog_destroy(msg_box);
|
|
CONTEXT_DATA(context)->posted_msg_box = NULL;
|
|
/* this is here so that we know when the file saying is over */
|
|
CONTEXT_DATA(context)->is_file_saving = False;
|
|
}
|
|
|
|
void
|
|
FE_ImageLoadDialog(MWContext* context)
|
|
{
|
|
Widget msg_box;
|
|
|
|
if (CONTEXT_DATA(context)->posted_msg_box != NULL)
|
|
return;
|
|
|
|
msg_box = fe_save_dialog_create(context, fe_SaveDialogImageLoadName);
|
|
|
|
fe_save_dialog_info_init(msg_box, XFE_SAVE_DIALOG_IMAGELOAD, 1);
|
|
CONTEXT_DATA(context)->posted_msg_box = msg_box;
|
|
|
|
fe_save_dialog_info_set(msg_box, "", 1);
|
|
|
|
XtManageChild(msg_box);
|
|
}
|
|
|
|
void
|
|
FE_ImageLoadDialogDestroy(MWContext* context)
|
|
{
|
|
Widget msg_box = CONTEXT_DATA(context)->posted_msg_box;
|
|
|
|
if (msg_box == NULL)
|
|
return;
|
|
|
|
fe_save_dialog_destroy(msg_box);
|
|
|
|
CONTEXT_DATA(context)->posted_msg_box = NULL;
|
|
}
|
|
|
|
typedef struct {
|
|
Widget form;
|
|
Widget widget;
|
|
unsigned response;
|
|
Boolean done;
|
|
} YesToAllInfo;
|
|
|
|
static void
|
|
fe_yes_to_all_cb(Widget widget, XtPointer closure, XtPointer call_data)
|
|
{
|
|
YesToAllInfo* info = (YesToAllInfo*)closure;
|
|
int i;
|
|
char* name;
|
|
char* widget_name = XtName(widget);
|
|
|
|
for (i = 0; (name = fe_CreateYesToAllDialog_button_data[i].name); i++) {
|
|
if (strcmp(name, widget_name) == 0) {
|
|
info->response = fe_CreateYesToAllDialog_button_data[i].type;
|
|
break;
|
|
}
|
|
}
|
|
|
|
info->widget = widget;
|
|
info->done = TRUE;
|
|
|
|
XtUnmanageChild(info->form);
|
|
}
|
|
|
|
#if 0
|
|
static unsigned
|
|
fe_OverWriteFileQuestionDialog(MWContext* context, char* filename)
|
|
{
|
|
char* msg;
|
|
int size;
|
|
Bool rv = FALSE;
|
|
|
|
size = XP_STRLEN(filename) + 1 /* slash*/
|
|
+ XP_STRLEN(fe_globalData.overwrite_file_message) + 1; /*nul*/
|
|
|
|
msg = (char *)XP_ALLOC(size);
|
|
if (!msg)
|
|
return False;
|
|
|
|
PR_snprintf(msg, size, fe_globalData.overwrite_file_message, filename);
|
|
|
|
if (FE_Confirm(context, msg))
|
|
rv = TRUE;
|
|
|
|
XP_FREE(msg);
|
|
|
|
return rv;
|
|
}
|
|
#endif
|
|
|
|
static unsigned
|
|
fe_DoYesToAllDialog(MWContext* context, char* filename)
|
|
{
|
|
YesToAllInfo info;
|
|
Arg av[20];
|
|
Cardinal ac;
|
|
XtCallbackRec cb_data;
|
|
XmString msg_string;
|
|
char* msg;
|
|
int size;
|
|
Widget form;
|
|
Widget default_button;
|
|
|
|
size = XP_STRLEN(filename) + 1 /* slash*/
|
|
+ XP_STRLEN(fe_globalData.overwrite_file_message) + 1; /*nul*/
|
|
|
|
msg = (char *)XP_ALLOC(size);
|
|
if (!msg)
|
|
return XFE_DIALOG_CANCEL_BUTTON;
|
|
|
|
PR_snprintf(msg, size, fe_globalData.overwrite_file_message, filename);
|
|
msg_string = XmStringCreateLocalized(msg);
|
|
XP_FREE(msg);
|
|
|
|
cb_data.callback = fe_yes_to_all_cb;
|
|
#ifdef OSF1
|
|
cb_data.closure = (void *)&info;
|
|
#else
|
|
cb_data.closure = &info;
|
|
#endif
|
|
|
|
ac = 0;
|
|
XtSetArg(av[ac], XmNarmCallback, &cb_data); ac++;
|
|
XtSetArg(av[ac], XmNmessageString, msg_string); ac++;
|
|
form = fe_CreateYesToAllDialog(context, "confirmSaveFiles", av, ac);
|
|
|
|
default_button = fe_YesToAllDialogGetChild(form, XFE_DAILOG_YESTOALL_BUTTON);
|
|
ac = 0;
|
|
XtSetArg(av[ac], XmNdefaultButton, default_button); ac++;
|
|
#if 0 /* why doesn't this work? */
|
|
XtSetArg(av[ac], XmNinitialFocus, default_button); ac++;
|
|
#endif
|
|
XtSetValues(form, av, ac);
|
|
|
|
info.response = XFE_DIALOG_CANCEL_BUTTON;
|
|
info.done = FALSE;
|
|
info.widget = NULL;
|
|
info.form = form;
|
|
|
|
XmStringFree(msg_string);
|
|
|
|
XtManageChild(form);
|
|
|
|
fe_NukeBackingStore(form);
|
|
while (!info.done)
|
|
fe_EventLoop();
|
|
|
|
XtDestroyWidget(form);
|
|
|
|
return info.response;
|
|
}
|
|
|
|
ED_SaveOption
|
|
FE_SaveFileExistsDialog(MWContext* context, char* filename)
|
|
{
|
|
switch (fe_DoYesToAllDialog(context, filename)) {
|
|
case XFE_DIALOG_YES_BUTTON:
|
|
return ED_SAVE_OVERWRITE_THIS;
|
|
case XFE_DAILOG_YESTOALL_BUTTON:
|
|
return ED_SAVE_OVERWRITE_ALL;
|
|
case XFE_DIALOG_NO_BUTTON:
|
|
return ED_SAVE_DONT_OVERWRITE_THIS;
|
|
case XFE_DIALOG_NOTOALL_BUTTON:
|
|
return ED_SAVE_DONT_OVERWRITE_ALL;
|
|
default:
|
|
return ED_SAVE_CANCEL;
|
|
}
|
|
}
|
|
|
|
Boolean
|
|
FE_SaveErrorContinueDialog(MWContext* context,
|
|
char* filename,
|
|
ED_FileError error)
|
|
{
|
|
if (error != ED_ERROR_NONE) {
|
|
/* There was a an error while saving on:\n%s */
|
|
/* The error code = (%d).*/
|
|
return fe_editor_report_file_error(context, error, filename, TRUE);
|
|
} else {
|
|
return TRUE;
|
|
}
|
|
}
|
|
|
|
Boolean XP_ConvertUrlToLocalFile (const char *url, char **localName)
|
|
{
|
|
Boolean filefound = FALSE;
|
|
char *path = NULL;
|
|
char *file = NULL;
|
|
XP_StatStruct statinfo;
|
|
|
|
if (localName)
|
|
{
|
|
*localName = NULL;
|
|
}
|
|
|
|
/* Must have "file:" url type and at least 1 character after '//' */
|
|
if (!url || !NET_IsLocalFileURL((char *) url))
|
|
{
|
|
return FALSE;
|
|
}
|
|
|
|
/* Extract file path from url: e.g. "/cl/foo/file.html" */
|
|
path = NET_ParseURL (url, GET_PATH_PART);
|
|
if (!path || XP_STRLEN(path) < 1) {
|
|
return FALSE;
|
|
}
|
|
|
|
#if 0
|
|
/* Get filename - skip over initial "/" */
|
|
file = WH_FileName (path+1, xpURL);
|
|
XP_FREE (path);
|
|
#endif
|
|
file = XP_STRDUP(path);
|
|
|
|
/* Test if file exists */
|
|
if (XP_Stat (file, &statinfo, xpURL) >= 0 &&
|
|
statinfo.st_mode & S_IFREG)
|
|
{
|
|
filefound = TRUE;
|
|
}
|
|
|
|
if (localName)
|
|
{
|
|
/* Pass string to caller; we didn't change it, so there's
|
|
no need to strdup
|
|
*/
|
|
*localName = file;
|
|
}
|
|
|
|
return filefound;
|
|
}
|
|
|
|
char *
|
|
XP_BackupFileName (const char *url)
|
|
{
|
|
char *filename = NULL;
|
|
char* path;
|
|
|
|
/* Must have a "file:" url and at least 1 character after the "//" */
|
|
if (!url || !NET_IsLocalFileURL((char *) url))
|
|
{
|
|
return FALSE;
|
|
}
|
|
|
|
/* Extract file path from url: e.g. "/cl/foo/file.html" */
|
|
path = NET_ParseURL (url, GET_PATH_PART);
|
|
if (!path || XP_STRLEN(path) < 1) {
|
|
return FALSE;
|
|
}
|
|
|
|
filename = (char *) XP_ALLOC ((XP_STRLEN (path) + 5) * sizeof (char));
|
|
if (!filename)
|
|
{
|
|
return NULL;
|
|
}
|
|
|
|
/* Get filename but ignore "file://" */
|
|
{
|
|
char* filename2 = WH_FileName (path, xpURL);
|
|
if (!filename2) return NULL;
|
|
XP_STRCPY (filename, filename2);
|
|
XP_FREE(filename2);
|
|
}
|
|
|
|
/* Add extension to the filename */
|
|
XP_STRCAT (filename, "~");
|
|
return filename;
|
|
}
|
|
|
|
void
|
|
fe_EditorUpdateToolbar(MWEditorContext* context, ED_TextFormat unused_for_now)
|
|
{
|
|
}
|
|
|
|
Boolean
|
|
fe_EditorLineBreakCanInsert(MWContext* context)
|
|
{
|
|
return (EDT_IsSelected(context) == FALSE);
|
|
}
|
|
|
|
/*
|
|
* type description
|
|
* ED_BREAK_NORMAL insert line break, ignore images.
|
|
* ED_BREAK_LEFT Break so it passes the image on the left
|
|
* ED_BREAK_RIGHT Break past the right image.
|
|
* ED_BREAK_BOTH Break below image(s).
|
|
*/
|
|
void
|
|
fe_EditorLineBreak(MWContext* context, ED_BreakType type)
|
|
{
|
|
if (!fe_EditorLineBreakCanInsert(context))
|
|
return;
|
|
EDT_InsertBreak(context, type);
|
|
fe_EditorUpdateToolbar(context, 0);
|
|
}
|
|
|
|
void
|
|
fe_EditorNonBreakingSpace(MWContext* context)
|
|
{
|
|
EDT_InsertNonbreakingSpace(context);
|
|
fe_EditorUpdateToolbar(context, 0);
|
|
}
|
|
|
|
void
|
|
fe_EditorParagraphMarksSetState(MWContext* context, Boolean display)
|
|
{
|
|
EDT_SetDisplayParagraphMarks(context, (Bool)display);
|
|
}
|
|
|
|
Boolean
|
|
fe_EditorParagraphMarksGetState(MWContext* context)
|
|
{
|
|
return (Boolean)EDT_GetDisplayParagraphMarks(context);
|
|
}
|
|
|
|
char*
|
|
fe_EditorDefaultGetTemplate()
|
|
{
|
|
return XFE_EDITOR_DEFAULT_DOCUMENT_TEMPLATE_URL;
|
|
}
|
|
|
|
void
|
|
fe_EditorDefaultGetColors(LO_Color* bg_color,
|
|
LO_Color* normal_color,
|
|
LO_Color* link_color,
|
|
LO_Color* active_color,
|
|
LO_Color* followed_color)
|
|
{
|
|
if (bg_color)
|
|
*bg_color = lo_master_colors[LO_COLOR_BG];
|
|
|
|
if (normal_color)
|
|
*normal_color = lo_master_colors[LO_COLOR_FG];
|
|
|
|
if (link_color)
|
|
*link_color = lo_master_colors[LO_COLOR_LINK];
|
|
|
|
if (active_color)
|
|
*active_color = lo_master_colors[LO_COLOR_ALINK];
|
|
|
|
if (followed_color)
|
|
*followed_color = lo_master_colors[LO_COLOR_VLINK];
|
|
}
|
|
|
|
static char* fe_last_publish_location;
|
|
static char* fe_last_publish_username;
|
|
static char* fe_last_publish_password;
|
|
|
|
#define SAFE_STRDUP(x) ((x)? XP_STRDUP(x): NULL)
|
|
|
|
Boolean
|
|
fe_EditorDefaultGetLastPublishLocation(MWContext* context,
|
|
char** location,
|
|
char** username,
|
|
char** password)
|
|
{
|
|
if (location)
|
|
*location = SAFE_STRDUP(fe_last_publish_location);
|
|
if (username)
|
|
*username = SAFE_STRDUP(fe_last_publish_username);
|
|
if (password)
|
|
*password = SECNAV_UnMungeString(fe_last_publish_password);
|
|
|
|
return fe_last_publish_password != NULL;
|
|
}
|
|
|
|
void
|
|
fe_EditorDefaultSetLastPublishLocation(MWContext* context,
|
|
char* location,
|
|
char* username,
|
|
char* password)
|
|
{
|
|
/* NOTE: this function is not MT safe */
|
|
if (fe_last_publish_location)
|
|
XP_FREE(fe_last_publish_location);
|
|
if (fe_last_publish_username)
|
|
XP_FREE(fe_last_publish_username);
|
|
if (fe_last_publish_password)
|
|
XP_FREE(fe_last_publish_password);
|
|
|
|
fe_last_publish_location = NULL;
|
|
fe_last_publish_username = NULL;
|
|
fe_last_publish_password = NULL;
|
|
|
|
if (location) {
|
|
char* full_location;
|
|
int rv;
|
|
|
|
fe_last_publish_location = XP_STRDUP(location);
|
|
|
|
if (username)
|
|
fe_last_publish_username = XP_STRDUP(username);
|
|
|
|
if (password)
|
|
fe_last_publish_password = SECNAV_MungeString(password);
|
|
|
|
full_location = NULL; /* having to do this is SO stupid */
|
|
rv = NET_MakeUploadURL(&full_location, location, username, NULL);
|
|
|
|
if (rv && full_location != NULL) {
|
|
PREF_SetCharPref("editor.publish_last_loc", full_location);
|
|
PREF_SetCharPref("editor.publish_last_pass",
|
|
fe_last_publish_password?
|
|
fe_last_publish_password: "");
|
|
}
|
|
}
|
|
}
|
|
|
|
void
|
|
fe_EditorPreferencesGetAutoSave(MWContext* context,
|
|
Boolean* enable,
|
|
unsigned* time)
|
|
{
|
|
if (fe_globalPrefs.editor_autosave_period == 0) {
|
|
*time = 0;
|
|
*enable = FALSE;
|
|
} else {
|
|
*time = fe_globalPrefs.editor_autosave_period;
|
|
*enable = TRUE;
|
|
}
|
|
}
|
|
|
|
void
|
|
fe_EditorPreferencesSetAutoSave(MWContext* context,
|
|
Boolean enable,
|
|
unsigned time)
|
|
{
|
|
struct fe_MWContext_cons *rest;
|
|
|
|
if (enable) {
|
|
fe_globalPrefs.editor_autosave_period = time;
|
|
} else {
|
|
fe_globalPrefs.editor_autosave_period = 0;
|
|
time = 0;
|
|
}
|
|
|
|
/*
|
|
* Walk over the contexts, and tell the BE to reset the
|
|
* autosave.
|
|
*/
|
|
for (rest = fe_all_MWContexts; rest; rest = rest->next) {
|
|
if (rest->context->type == MWContextEditor) {
|
|
EDT_SetAutoSavePeriod(rest->context, time);
|
|
}
|
|
}
|
|
}
|
|
|
|
void
|
|
fe_EditorPreferencesGetLinksAndImages(MWContext* context,
|
|
Boolean* links,
|
|
Boolean* images)
|
|
{
|
|
if (links)
|
|
*links = fe_globalPrefs.editor_maintain_links;
|
|
if (images)
|
|
*images = fe_globalPrefs.editor_keep_images;
|
|
}
|
|
|
|
void
|
|
fe_EditorPreferencesSetLinksAndImages(MWContext* context,
|
|
Boolean links,
|
|
Boolean images)
|
|
{
|
|
fe_globalPrefs.editor_maintain_links = links;
|
|
fe_globalPrefs.editor_keep_images = images;
|
|
}
|
|
|
|
Boolean
|
|
fe_EditorPreferencesGetPublishLocation(MWContext* context,
|
|
char** location,
|
|
char** username,
|
|
char** password)
|
|
{
|
|
if (location)
|
|
*location = SAFE_STRDUP(fe_globalPrefs.editor_publish_location);
|
|
if (username)
|
|
*username = SAFE_STRDUP(fe_globalPrefs.editor_publish_username);
|
|
if (password)
|
|
*password =
|
|
SECNAV_UnMungeString(fe_globalPrefs.editor_publish_password);
|
|
|
|
return (fe_globalPrefs.editor_publish_password != NULL);
|
|
}
|
|
|
|
void
|
|
fe_EditorPreferencesSetPublishLocation(MWContext* context,
|
|
char* location,
|
|
char* username,
|
|
char* password)
|
|
{
|
|
if (fe_globalPrefs.editor_publish_location)
|
|
XP_FREE(fe_globalPrefs.editor_publish_location);
|
|
if (fe_globalPrefs.editor_publish_username)
|
|
XP_FREE(fe_globalPrefs.editor_publish_username);
|
|
if (fe_globalPrefs.editor_publish_password)
|
|
XP_FREE(fe_globalPrefs.editor_publish_password);
|
|
|
|
fe_globalPrefs.editor_publish_location = NULL;
|
|
fe_globalPrefs.editor_publish_username = NULL;
|
|
fe_globalPrefs.editor_publish_password = NULL;
|
|
|
|
if (location) {
|
|
fe_globalPrefs.editor_publish_location = XP_STRDUP(location);
|
|
|
|
if (username)
|
|
fe_globalPrefs.editor_publish_username = XP_STRDUP(username);
|
|
|
|
if (password)
|
|
fe_globalPrefs.editor_publish_password
|
|
= SECNAV_MungeString(password);
|
|
}
|
|
fe_globalPrefs.editor_save_publish_password = (password != NULL);
|
|
}
|
|
|
|
char*
|
|
fe_EditorPreferencesGetBrowseLocation(MWContext* context)
|
|
{
|
|
return SAFE_STRDUP(fe_globalPrefs.editor_browse_location);
|
|
}
|
|
|
|
void
|
|
fe_EditorPreferencesSetBrowseLocation(MWContext* context, char* location)
|
|
{
|
|
if (fe_globalPrefs.editor_browse_location)
|
|
XP_FREE(fe_globalPrefs.editor_browse_location);
|
|
fe_globalPrefs.editor_browse_location = NULL;
|
|
|
|
if (location)
|
|
fe_globalPrefs.editor_browse_location = XP_STRDUP(location);
|
|
}
|
|
|
|
char*
|
|
fe_EditorPreferencesGetAuthor(MWContext* context)
|
|
{
|
|
if (fe_globalPrefs.editor_author_name)
|
|
return fe_globalPrefs.editor_author_name;
|
|
else
|
|
return "";
|
|
}
|
|
|
|
void
|
|
fe_EditorPreferencesSetAuthor(MWContext* context, char* name)
|
|
{
|
|
if (fe_globalPrefs.editor_author_name)
|
|
XtFree(fe_globalPrefs.editor_author_name);
|
|
fe_globalPrefs.editor_author_name = XtNewString(name);
|
|
}
|
|
|
|
char*
|
|
fe_EditorPreferencesGetTemplate(MWContext* context)
|
|
{
|
|
if (fe_globalPrefs.editor_document_template)
|
|
return fe_globalPrefs.editor_document_template;
|
|
else
|
|
return NULL;
|
|
}
|
|
|
|
void
|
|
fe_EditorPreferencesSetTemplate(MWContext* context, char* name)
|
|
{
|
|
if (fe_globalPrefs.editor_document_template)
|
|
XtFree(fe_globalPrefs.editor_document_template);
|
|
fe_globalPrefs.editor_document_template = XtNewString(name);
|
|
}
|
|
|
|
void
|
|
fe_EditorPreferencesGetEditors(MWContext* context, char** html, char** image)
|
|
{
|
|
if (html != NULL) {
|
|
if (fe_globalPrefs.editor_html_editor)
|
|
*html = fe_globalPrefs.editor_html_editor;
|
|
else
|
|
*html = NULL;
|
|
}
|
|
|
|
if (image != NULL) {
|
|
if (fe_globalPrefs.editor_image_editor)
|
|
*image = fe_globalPrefs.editor_image_editor;
|
|
else
|
|
*image = NULL;
|
|
}
|
|
}
|
|
|
|
void
|
|
fe_EditorPreferencesSetEditors(MWContext* context, char* html, char* image)
|
|
{
|
|
if (html != NULL) {
|
|
if (fe_globalPrefs.editor_html_editor)
|
|
XtFree(fe_globalPrefs.editor_html_editor);
|
|
fe_globalPrefs.editor_html_editor = XtNewString(html);
|
|
}
|
|
|
|
if (image != NULL) {
|
|
if (fe_globalPrefs.editor_image_editor)
|
|
XtFree(fe_globalPrefs.editor_image_editor);
|
|
fe_globalPrefs.editor_image_editor = XtNewString(image);
|
|
}
|
|
}
|
|
|
|
Boolean
|
|
fe_EditorPreferencesGetColors(MWContext* context,
|
|
char* background_image,
|
|
LO_Color* bg_color,
|
|
LO_Color* normal_color,
|
|
LO_Color* link_color,
|
|
LO_Color* active_color,
|
|
LO_Color* followed_color)
|
|
{
|
|
if (bg_color)
|
|
*bg_color = fe_globalPrefs.editor_background_color;
|
|
if (normal_color)
|
|
*normal_color = fe_globalPrefs.editor_normal_color;
|
|
if (link_color)
|
|
*link_color = fe_globalPrefs.editor_link_color;
|
|
if (active_color)
|
|
*active_color = fe_globalPrefs.editor_active_color;
|
|
if (followed_color)
|
|
*followed_color = fe_globalPrefs.editor_followed_color;
|
|
|
|
if (background_image) {
|
|
if (fe_globalPrefs.editor_background_image)
|
|
strcpy(background_image, fe_globalPrefs.editor_background_image);
|
|
else
|
|
background_image[0] = '\0';
|
|
}
|
|
|
|
return fe_globalPrefs.editor_custom_colors;
|
|
}
|
|
|
|
void
|
|
fe_EditorPreferencesSetColors(MWContext* context,
|
|
char* background_image,
|
|
LO_Color* bg_color,
|
|
LO_Color* normal_color,
|
|
LO_Color* link_color,
|
|
LO_Color* active_color,
|
|
LO_Color* followed_color)
|
|
{
|
|
XP_Bool set = FALSE;
|
|
|
|
if (bg_color) {
|
|
fe_globalPrefs.editor_background_color = *bg_color;
|
|
set = TRUE;
|
|
}
|
|
if (normal_color) {
|
|
fe_globalPrefs.editor_normal_color = *normal_color;
|
|
set = TRUE;
|
|
}
|
|
if (link_color) {
|
|
fe_globalPrefs.editor_link_color = *link_color;
|
|
set = TRUE;
|
|
}
|
|
if (active_color) {
|
|
fe_globalPrefs.editor_active_color = *active_color;
|
|
set = TRUE;
|
|
}
|
|
if (followed_color) {
|
|
fe_globalPrefs.editor_followed_color = *followed_color;
|
|
set = TRUE;
|
|
}
|
|
|
|
fe_globalPrefs.editor_custom_colors = set;
|
|
|
|
if (fe_globalPrefs.editor_background_image)
|
|
XtFree(fe_globalPrefs.editor_background_image);
|
|
if (background_image && background_image[0] != '\0')
|
|
fe_globalPrefs.editor_background_image = XtNewString(background_image);
|
|
else
|
|
fe_globalPrefs.editor_background_image = NULL;
|
|
}
|
|
|
|
Boolean
|
|
fe_EditorDocumentGetImagesWithDocument(MWContext* context)
|
|
{
|
|
Boolean keep;
|
|
EDT_PageData* page_data = EDT_GetPageData(context);
|
|
|
|
if (page_data == NULL) {
|
|
fe_EditorPreferencesGetLinksAndImages(context, NULL, &keep);
|
|
} else {
|
|
keep = page_data->bKeepImagesWithDoc;
|
|
EDT_FreePageData(page_data);
|
|
}
|
|
|
|
return keep;
|
|
}
|
|
|
|
void
|
|
fe_EditorDocumentSetImagesWithDocument(MWContext* context, Boolean keep)
|
|
{
|
|
EDT_PageData* page_data = EDT_GetPageData(context);
|
|
|
|
if (page_data != NULL) {
|
|
page_data->bKeepImagesWithDoc = keep;
|
|
EDT_SetPageData(context, page_data);
|
|
EDT_FreePageData(page_data);
|
|
}
|
|
}
|
|
|
|
char*
|
|
fe_EditorDocumentLocationGet(MWContext* context)
|
|
{
|
|
History_entry* hist_ent;
|
|
|
|
/* get the location */
|
|
hist_ent = SHIST_GetCurrent(&context->hist);
|
|
if (hist_ent)
|
|
return hist_ent->address;
|
|
else
|
|
return NULL;
|
|
}
|
|
|
|
void
|
|
fe_EditorDocumentSetTitle(MWContext* context, char* name)
|
|
{
|
|
EDT_PageData* page_data = EDT_GetPageData(context);
|
|
|
|
if (!page_data)
|
|
return;
|
|
|
|
if (page_data->pTitle)
|
|
XP_FREE(page_data->pTitle);
|
|
|
|
if (name && name[0] != '\0')
|
|
page_data->pTitle = XP_STRDUP(name);
|
|
else
|
|
page_data->pTitle = NULL;
|
|
|
|
EDT_SetPageData(context, page_data);
|
|
|
|
EDT_FreePageData(page_data);
|
|
}
|
|
|
|
char*
|
|
fe_EditorDocumentGetTitle(MWContext* context)
|
|
{
|
|
EDT_PageData* page_data;
|
|
char* value = NULL;
|
|
|
|
/* get the title */
|
|
page_data = EDT_GetPageData(context);
|
|
if (page_data != NULL) {
|
|
if (page_data->pTitle != NULL && page_data->pTitle[0] != '\0')
|
|
value = XP_STRDUP(page_data->pTitle);
|
|
|
|
EDT_FreePageData(page_data);
|
|
}
|
|
|
|
return value;
|
|
}
|
|
|
|
#define BEGIN_DIRTY(xxx) XP_Bool dirty = FALSE
|
|
|
|
#define MARK_DIRTY(context) \
|
|
if (!(dirty)) { \
|
|
EDT_BeginBatchChanges((context)); \
|
|
dirty = TRUE; \
|
|
}
|
|
|
|
#define END_DIRTY(context) \
|
|
if ((dirty)) { \
|
|
EDT_EndBatchChanges((context)); \
|
|
}
|
|
|
|
void
|
|
fe_EditorDocumentSetMetaData(MWContext* context, char* name, char* value)
|
|
{
|
|
EDT_MetaData *pData = EDT_NewMetaData();
|
|
|
|
BEGIN_DIRTY(context);
|
|
|
|
if ( pData ) {
|
|
|
|
pData->bHttpEquiv = FALSE;
|
|
if ( name && XP_STRLEN(name) > 0 ) {
|
|
|
|
MARK_DIRTY(context);
|
|
|
|
pData->pName = XP_STRDUP(name);
|
|
if ( value && XP_STRLEN(value) > 0 ) {
|
|
pData->pContent = XP_STRDUP(value);
|
|
EDT_SetMetaData(context, pData);
|
|
} else {
|
|
/* (Don't really need to do this) */
|
|
pData->pContent = NULL;
|
|
/* Remove the item */
|
|
EDT_DeleteMetaData(context, pData);
|
|
}
|
|
}
|
|
|
|
EDT_FreeMetaData(pData);
|
|
}
|
|
|
|
END_DIRTY(context);
|
|
}
|
|
|
|
char*
|
|
fe_EditorDocumentGetMetaData(MWContext* context, char* name)
|
|
{
|
|
EDT_MetaData* pData;
|
|
int count;
|
|
int i;
|
|
|
|
/* lifted from winfe/edprops.cpp */
|
|
count = EDT_MetaDataCount(context);
|
|
for (i = 0; i < count; i++ ) {
|
|
pData = EDT_GetMetaData(context, i);
|
|
if (pData != NULL && !pData->bHttpEquiv) {
|
|
if (XP_STRCASECMP(pData->pName, name) == 0) {
|
|
return pData->pContent;
|
|
}
|
|
}
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
static char* fe_editor_known_meta_data_names[] = {
|
|
"Author",
|
|
"Description",
|
|
#if 0 /* this one need to end up in user variables */
|
|
"Generator",
|
|
#endif
|
|
"Last-Modified",
|
|
"Created",
|
|
"Classification",
|
|
"Keywords",
|
|
NULL
|
|
};
|
|
|
|
static Boolean
|
|
fe_is_known_meta_data_name(char* name)
|
|
{
|
|
unsigned i;
|
|
|
|
for (i = 0; fe_editor_known_meta_data_names[i] != NULL; i++) {
|
|
if (XP_STRCASECMP(fe_editor_known_meta_data_names[i], name) == 0)
|
|
return TRUE;
|
|
}
|
|
return FALSE;
|
|
}
|
|
|
|
fe_NameValueItem*
|
|
fe_EditorDocumentGetHttpEquivMetaDataList(MWContext* context)
|
|
{
|
|
int count = EDT_MetaDataCount(context);
|
|
int i;
|
|
EDT_MetaData* pData;
|
|
fe_NameValueItem* items;
|
|
unsigned nitems = 0;
|
|
|
|
items = (fe_NameValueItem*)XtMalloc(sizeof(fe_NameValueItem)*(count+1));
|
|
|
|
for (i = 0; i < count; i++) {
|
|
pData = EDT_GetMetaData(context, i);
|
|
|
|
if (pData != NULL && pData->bHttpEquiv != 0) {
|
|
items[nitems].name = XtNewString(pData->pName);
|
|
items[nitems].value = XtNewString(pData->pContent);
|
|
nitems++;
|
|
}
|
|
}
|
|
items[nitems].name = NULL;
|
|
items[nitems].value = NULL;
|
|
|
|
items = (fe_NameValueItem*)XtRealloc((XtPointer)items,
|
|
sizeof(fe_NameValueItem)*(nitems+1));
|
|
|
|
return items;
|
|
}
|
|
|
|
void
|
|
fe_EditorDocumentSetHttpEquivMetaDataList(MWContext* context,
|
|
fe_NameValueItem* list)
|
|
{
|
|
fe_NameValueItem* old_list;
|
|
int i;
|
|
int j;
|
|
EDT_MetaData *meta_data;
|
|
BEGIN_DIRTY(context);
|
|
|
|
old_list = fe_EditorDocumentGetHttpEquivMetaDataList(context);
|
|
|
|
/* first delete items no longer in set */
|
|
for (i = 0; old_list[i].name != NULL; i++) {
|
|
|
|
for (j = 0; list[j].name != NULL; j++) {
|
|
if (XP_STRCASECMP(old_list[i].name, list[j].name) == 0) /* match */
|
|
break;
|
|
}
|
|
|
|
/* I'm sure all this thrashing of the heap can't be required ?? */
|
|
if (list[j].name == NULL) { /* it's gone now */
|
|
meta_data = EDT_NewMetaData();
|
|
meta_data->pName = old_list[i].name;
|
|
meta_data->pContent = NULL;
|
|
MARK_DIRTY(context);
|
|
EDT_DeleteMetaData(context, meta_data);
|
|
EDT_FreeMetaData(meta_data);
|
|
}
|
|
}
|
|
|
|
/* now set the ones that have changed */
|
|
for (j = 0; list[j].name != NULL; j++) {
|
|
meta_data = EDT_NewMetaData();
|
|
meta_data->pName = list[j].name;
|
|
meta_data->pContent = list[j].value;
|
|
meta_data->bHttpEquiv = TRUE;
|
|
MARK_DIRTY(context);
|
|
EDT_SetMetaData(context, meta_data);
|
|
EDT_FreeMetaData(meta_data);
|
|
}
|
|
|
|
END_DIRTY(context);
|
|
}
|
|
|
|
fe_NameValueItem*
|
|
fe_EditorDocumentGetAdvancedMetaDataList(MWContext* context)
|
|
{
|
|
int count = EDT_MetaDataCount(context);
|
|
int i;
|
|
EDT_MetaData* pData;
|
|
fe_NameValueItem* items;
|
|
unsigned nitems = 0;
|
|
|
|
items = (fe_NameValueItem*)XtMalloc(sizeof(fe_NameValueItem)*(count+1));
|
|
|
|
for (i = 0; i < count; i++) {
|
|
pData = EDT_GetMetaData(context, i);
|
|
|
|
if (pData == NULL)
|
|
continue;
|
|
|
|
/* is it one of the ones handled in document general */
|
|
if ((pData->bHttpEquiv) || fe_is_known_meta_data_name(pData->pName))
|
|
continue;
|
|
|
|
items[nitems].name = XtNewString(pData->pName);
|
|
items[nitems].value = XtNewString(pData->pContent);
|
|
nitems++;
|
|
}
|
|
items[nitems].name = NULL;
|
|
items[nitems].value = NULL;
|
|
|
|
items = (fe_NameValueItem*)XtRealloc((XtPointer)items,
|
|
sizeof(fe_NameValueItem)*(nitems+1));
|
|
|
|
return items;
|
|
}
|
|
|
|
void
|
|
fe_EditorDocumentSetAdvancedMetaDataList(MWContext* context,
|
|
fe_NameValueItem* list)
|
|
{
|
|
fe_NameValueItem* old_list;
|
|
int i;
|
|
int j;
|
|
EDT_MetaData *meta_data;
|
|
BEGIN_DIRTY(context);
|
|
|
|
old_list = fe_EditorDocumentGetAdvancedMetaDataList(context);
|
|
|
|
/* first delete items no longer in set */
|
|
for (i = 0; old_list[i].name != NULL; i++) {
|
|
|
|
for (j = 0; list[j].name != NULL; j++) {
|
|
if (XP_STRCASECMP(old_list[i].name, list[j].name) == 0) /* match */
|
|
break;
|
|
}
|
|
|
|
/* I'm sure all this thrashing of the heap can't be required ?? */
|
|
if (list[j].name == NULL) { /* it's gone now */
|
|
meta_data = EDT_NewMetaData();
|
|
meta_data->pName = old_list[i].name;
|
|
meta_data->pContent = NULL;
|
|
MARK_DIRTY(context);
|
|
EDT_DeleteMetaData(context, meta_data);
|
|
EDT_FreeMetaData(meta_data);
|
|
}
|
|
}
|
|
|
|
/* now set the ones that have changed */
|
|
for (j = 0; list[j].name != NULL; j++) {
|
|
meta_data = EDT_NewMetaData();
|
|
meta_data->pName = list[j].name;
|
|
meta_data->pContent = list[j].value;
|
|
MARK_DIRTY(context);
|
|
EDT_SetMetaData(context, meta_data);
|
|
EDT_FreeMetaData(meta_data);
|
|
}
|
|
|
|
END_DIRTY(context);
|
|
}
|
|
|
|
Boolean
|
|
fe_EditorDocumentGetColors(MWContext* context,
|
|
char* background_image,
|
|
Boolean* leave_image,
|
|
LO_Color* bg_color,
|
|
LO_Color* normal_color,
|
|
LO_Color* link_color,
|
|
LO_Color* active_color,
|
|
LO_Color* followed_color)
|
|
{
|
|
EDT_PageData* page_data = EDT_GetPageData(context);
|
|
Boolean set = FALSE;
|
|
|
|
fe_EditorDefaultGetColors(bg_color,
|
|
normal_color,
|
|
link_color,
|
|
active_color,
|
|
followed_color);
|
|
|
|
if (!page_data)
|
|
return FALSE;
|
|
|
|
if (background_image != NULL) {
|
|
if (page_data->pBackgroundImage != NULL) {
|
|
XP_STRCPY(background_image, page_data->pBackgroundImage);
|
|
} else {
|
|
background_image[0] = '\0';
|
|
}
|
|
}
|
|
|
|
if (bg_color) {
|
|
if (page_data->pColorBackground) {
|
|
*bg_color = *page_data->pColorBackground;
|
|
set = TRUE;
|
|
}
|
|
}
|
|
|
|
if (normal_color) {
|
|
if (page_data->pColorText) {
|
|
*normal_color = *page_data->pColorText;
|
|
set = TRUE;
|
|
}
|
|
}
|
|
|
|
if (link_color) {
|
|
if (page_data->pColorLink) {
|
|
*link_color = *page_data->pColorLink;
|
|
set = TRUE;
|
|
}
|
|
}
|
|
|
|
if (active_color) {
|
|
if (page_data->pColorActiveLink) {
|
|
*active_color = *page_data->pColorActiveLink;
|
|
set = TRUE;
|
|
}
|
|
}
|
|
|
|
if (followed_color) {
|
|
if (page_data->pColorFollowedLink) {
|
|
*followed_color = *page_data->pColorFollowedLink;
|
|
set = TRUE;
|
|
}
|
|
}
|
|
|
|
if (leave_image) {
|
|
*leave_image = page_data->bBackgroundNoSave;
|
|
}
|
|
|
|
|
|
EDT_FreePageData(page_data);
|
|
|
|
return set;
|
|
}
|
|
|
|
void
|
|
fe_EditorDocumentSetColors(MWContext* context,
|
|
char* background_image,
|
|
Boolean leave_image,
|
|
LO_Color* bg_color,
|
|
LO_Color* normal_color,
|
|
LO_Color* link_color,
|
|
LO_Color* active_color,
|
|
LO_Color* followed_color)
|
|
{
|
|
EDT_PageData save_data;
|
|
EDT_PageData* page_data;
|
|
|
|
if (background_image != NULL && background_image[0] == '\0')
|
|
background_image = NULL;
|
|
|
|
page_data = EDT_GetPageData(context);
|
|
|
|
if (!page_data)
|
|
return;
|
|
|
|
memcpy(&save_data, page_data, sizeof(EDT_PageData));
|
|
|
|
page_data->pBackgroundImage = background_image;
|
|
page_data->pColorBackground = bg_color;
|
|
page_data->pColorText = normal_color;
|
|
page_data->pColorLink = link_color;
|
|
page_data->pColorActiveLink = active_color;
|
|
page_data->pColorFollowedLink = followed_color;
|
|
page_data->bBackgroundNoSave = leave_image;
|
|
|
|
EDT_SetPageData(context, page_data);
|
|
memcpy(page_data, &save_data, sizeof(EDT_PageData));
|
|
EDT_FreePageData(page_data);
|
|
}
|
|
|
|
typedef struct fe_EditorCommandDescription {
|
|
intn id;
|
|
char* name;
|
|
} fe_EditorCommandDescription;
|
|
|
|
/*
|
|
* Yuck lifted from editor.h
|
|
*/
|
|
#define kNullCommandID 0
|
|
#define kTypingCommandID 1
|
|
#define kAddTextCommandID 2
|
|
#define kDeleteTextCommandID 3
|
|
#define kCutCommandID 4
|
|
#define kPasteTextCommandID 5
|
|
#define kPasteHTMLCommandID 6
|
|
#define kPasteHREFCommandID 7
|
|
#define kChangeAttributesCommandID 8
|
|
#define kIndentCommandID 9
|
|
#define kParagraphAlignCommandID 10
|
|
#define kMorphContainerCommandID 11
|
|
#define kInsertHorizRuleCommandID 12
|
|
#define kSetHorizRuleDataCommandID 13
|
|
#define kInsertImageCommandID 14
|
|
#define kSetImageDataCommandID 15
|
|
#define kInsertBreakCommandID 16
|
|
#define kChangePageDataCommandID 17
|
|
#define kSetMetaDataCommandID 18
|
|
#define kDeleteMetaDataCommandID 19
|
|
#define kInsertTargetCommandID 20
|
|
#define kSetTargetDataCommandID 21
|
|
#define kInsertUnknownTagCommandID 22
|
|
#define kSetUnknownTagDataCommandID 23
|
|
#define kGroupOfChangesCommandID 24
|
|
#define kSetListDataCommandID 25
|
|
|
|
#define kInsertTableCommandID 26
|
|
#define kDeleteTableCommandID 27
|
|
#define kSetTableDataCommandID 28
|
|
|
|
#define kInsertTableCaptionCommandID 29
|
|
#define kSetTableCaptionDataCommandID 30
|
|
#define kDeleteTableCaptionCommandID 31
|
|
|
|
#define kInsertTableRowCommandID 32
|
|
#define kSetTableRowDataCommandID 33
|
|
#define kDeleteTableRowCommandID 34
|
|
|
|
#define kInsertTableColumnCommandID 35
|
|
#define kSetTableCellDataCommandID 36
|
|
#define kDeleteTableColumnCommandID 37
|
|
|
|
#define kInsertTableCellCommandID 38
|
|
#define kDeleteTableCellCommandID 39
|
|
|
|
static fe_EditorCommandDescription fe_editor_commands[] = {
|
|
{ kNullCommandID, "Null" },
|
|
{ kTypingCommandID, "Typing" },
|
|
{ kAddTextCommandID, "AddText" },
|
|
{ kDeleteTextCommandID, "DeleteText" },
|
|
{ kCutCommandID, "Cut" },
|
|
{ kPasteTextCommandID, "PasteText" },
|
|
{ kPasteHTMLCommandID, "PasteHTML" },
|
|
{ kPasteHREFCommandID, "PasteHREF" },
|
|
{ kChangeAttributesCommandID, "ChangeAttributes" },
|
|
{ kIndentCommandID, "Indent" },
|
|
{ kParagraphAlignCommandID, "ParagraphAlign" },
|
|
{ kMorphContainerCommandID, "MorphContainer" },
|
|
{ kInsertHorizRuleCommandID, "InsertHorizRule" },
|
|
{ kSetHorizRuleDataCommandID, "SetHorizRuleData" },
|
|
{ kInsertImageCommandID, "InsertImage" },
|
|
{ kSetImageDataCommandID, "SetImageData" },
|
|
{ kInsertBreakCommandID, "InsertBreak" },
|
|
{ kChangePageDataCommandID, "ChangePageData" },
|
|
{ kSetMetaDataCommandID, "SetMetaData" },
|
|
{ kDeleteMetaDataCommandID, "DeleteMetaData" },
|
|
{ kInsertTargetCommandID, "InsertTarget" },
|
|
{ kSetTargetDataCommandID, "SetTargetData" },
|
|
{ kInsertUnknownTagCommandID, "InsertUnknownTag" },
|
|
{ kSetUnknownTagDataCommandID,"SetUnknownTagData" },
|
|
{ kGroupOfChangesCommandID, "GroupOfChanges" },
|
|
{ kSetListDataCommandID, "SetListData" },
|
|
{ kInsertTableCommandID, "InsertTable" },
|
|
{ kDeleteTableCommandID, "DeleteTable" },
|
|
{ kSetTableDataCommandID, "SetTableData" },
|
|
{ kInsertTableCaptionCommandID, "InsertTableCaption" },
|
|
{ kSetTableCaptionDataCommandID, "SetTableCaptionData" },
|
|
{ kDeleteTableCaptionCommandID, "DeleteTableCaption" },
|
|
{ kInsertTableRowCommandID, "InsertTableRow" },
|
|
{ kSetTableRowDataCommandID, "SetTableRowData" },
|
|
{ kDeleteTableRowCommandID, "DeleteTableRow" },
|
|
{ kInsertTableColumnCommandID, "InsertTableColumn" },
|
|
{ kSetTableCellDataCommandID, "SetTableCellData" },
|
|
{ kDeleteTableColumnCommandID, "DeleteTableColumn" },
|
|
{ kInsertTableCellCommandID, "InsertTableCell" },
|
|
{ kDeleteTableCellCommandID, "DeleteTableCell" },
|
|
|
|
{ -1, NULL }
|
|
};
|
|
|
|
static XmString
|
|
resource_motif_string(Widget widget, char* name)
|
|
{
|
|
XmString result = XfeSubResourceGetXmStringValue(widget,
|
|
name,
|
|
"UndoItem",
|
|
XmNlabelString,
|
|
XmCXmString,
|
|
NULL);
|
|
|
|
if (!result)
|
|
result = XmStringCreateLocalized(name);
|
|
|
|
return result;
|
|
}
|
|
|
|
static XmString string_cache;
|
|
|
|
static XmString
|
|
fe_editor_undo_get_label(MWContext* context, intn id, Boolean is_undo)
|
|
{
|
|
char buf[64];
|
|
XmString string;
|
|
char* name;
|
|
int n;
|
|
|
|
name = NULL;
|
|
for (n = 0; fe_editor_commands[n].name != NULL; n++) {
|
|
if (fe_editor_commands[n].id == id) {
|
|
name = fe_editor_commands[n].name;
|
|
break;
|
|
}
|
|
}
|
|
|
|
/*
|
|
* For ids we don't know yet, just show "Undo" or "Redo"
|
|
*/
|
|
if (!name)
|
|
name = "";
|
|
|
|
/*
|
|
* Look for it in the cache here!!!!
|
|
*/
|
|
/*FIXME*/
|
|
if (string_cache)
|
|
XmStringFree(string_cache);
|
|
|
|
if (is_undo)
|
|
XP_STRCPY(buf, "undo");
|
|
else
|
|
XP_STRCPY(buf, "redo");
|
|
XP_STRCAT(buf, name);
|
|
|
|
/*
|
|
* Resource the string.
|
|
*/
|
|
/*
|
|
* Why is it so hard to get a string! Like, what,
|
|
* I'd never want to do that.
|
|
*/
|
|
string = resource_motif_string(CONTEXT_DATA(context)->menubar, buf);
|
|
string_cache = string;
|
|
|
|
return string;
|
|
}
|
|
|
|
Boolean
|
|
fe_EditorCanUndo(MWContext* context, XmString* label_return)
|
|
{
|
|
intn command_id = EDT_GetUndoCommandID(context, 0);
|
|
|
|
if (label_return) {
|
|
*label_return = fe_editor_undo_get_label(context,
|
|
command_id,
|
|
True);
|
|
}
|
|
|
|
if (command_id != CEDITCOMMAND_ID_NULL)
|
|
return True;
|
|
else
|
|
return FALSE;
|
|
}
|
|
|
|
Boolean
|
|
fe_EditorCanRedo(MWContext* context, XmString* label_return)
|
|
{
|
|
intn command_id = EDT_GetRedoCommandID(context, 0);
|
|
|
|
if (label_return) {
|
|
*label_return = fe_editor_undo_get_label(context,
|
|
command_id,
|
|
False);
|
|
}
|
|
|
|
if (command_id != CEDITCOMMAND_ID_NULL)
|
|
return True;
|
|
else
|
|
return FALSE;
|
|
}
|
|
|
|
void
|
|
fe_EditorUndo(MWContext* context)
|
|
{
|
|
if (EDT_GetUndoCommandID(context, 0) != CEDITCOMMAND_ID_NULL) {
|
|
EDT_Undo(context);
|
|
fe_EditorUpdateToolbar(context, 0);
|
|
} else {
|
|
fe_Bell(context);
|
|
}
|
|
}
|
|
|
|
void
|
|
fe_EditorRedo(MWContext* context)
|
|
{
|
|
if (EDT_GetRedoCommandID(context, 0) != CEDITCOMMAND_ID_NULL) {
|
|
EDT_Redo(context);
|
|
fe_EditorUpdateToolbar(context, 0);
|
|
} else {
|
|
fe_Bell(context);
|
|
}
|
|
}
|
|
|
|
Boolean
|
|
fe_EditorCanCut(MWContext* context)
|
|
{
|
|
return (EDT_COP_OK == EDT_CanCut(context, FALSE));
|
|
}
|
|
|
|
Boolean
|
|
fe_EditorCanCopy(MWContext* context)
|
|
{
|
|
return (EDT_COP_OK == EDT_CanCopy(context, TRUE));
|
|
}
|
|
|
|
static char fe_editor_data_name[] = "NETSCAPE_HTML";
|
|
static char fe_editor_html_name[] = "HTML";
|
|
static char fe_editor_text_name[] = "STRING";
|
|
static char fe_editor_app_name[] = "MOZILLA";
|
|
|
|
static Atom XFE_TEXT_ATOM = 0;
|
|
static Atom XFE_HTML_ATOM = 0;
|
|
static Atom XFE_DATA_ATOM = 0;
|
|
|
|
#define XFE_CCP_TEXT (fe_editor_text_name)
|
|
#define XFE_CCP_HTML (fe_editor_html_name)
|
|
#define XFE_CCP_DATA (fe_editor_data_name)
|
|
#define XFE_CCP_NAME (fe_editor_app_name)
|
|
|
|
#if 0
|
|
/*
|
|
* Cannot get this to work.
|
|
*/
|
|
static void
|
|
fe_property_notify_eh(Widget w, XtPointer closure, XEvent *ev, Boolean *cont)
|
|
{
|
|
MWContext* context = (MWContext *)closure;
|
|
Display* display = XtDisplay(CONTEXT_WIDGET(context));
|
|
Atom cb_atom = XmInternAtom(display, "CLIPBOARD", False);
|
|
Atom string_atom = XmInternAtom(display, XFE_CCP_TEXT, False);
|
|
Atom html_atom = XmInternAtom(display, XFE_CCP_HTML, False);
|
|
Atom data_atom = XmInternAtom(display, XFE_CCP_DATA, False);
|
|
Atom long_string_atom = XmInternAtom(display, "_MOTIF_CLIP_FORMAT_" "STRING", False);
|
|
Atom long_html_atom = XmInternAtom(display, "_MOTIF_CLIP_FORMAT_" "NETSCAPE_HTML", False);
|
|
fe_EditorContextData* editor;
|
|
|
|
#if 0
|
|
if (ev->xproperty.atom == XA_CUT_BUFFER0) {
|
|
editor = EDITOR_CONTEXT_DATA(context);
|
|
fe_editor_paste_update_cb(editor->toolbar_paste,
|
|
(XtPointer)context, NULL);
|
|
}
|
|
#endif
|
|
|
|
#if 0
|
|
if (ev->xproperty.atom == XmInternAtom (dpy, "CLIPBOARD", False)XA_CUT_BUFFER0)
|
|
#endif
|
|
}
|
|
#endif
|
|
|
|
Boolean
|
|
fe_EditorCanPasteData(MWContext* context, int32* r_length)
|
|
{
|
|
Widget mainw = CONTEXT_WIDGET(context);
|
|
Display* display = XtDisplay(mainw);
|
|
Window window = XtWindow(mainw);
|
|
int cb_result;
|
|
unsigned long length;
|
|
|
|
cb_result = XmClipboardInquireLength(display, window,
|
|
XFE_CCP_DATA, &length);
|
|
|
|
if (cb_result == ClipboardSuccess && length > 0) {
|
|
*r_length = length;
|
|
return TRUE;
|
|
} else {
|
|
*r_length = 0;
|
|
return FALSE;
|
|
}
|
|
}
|
|
|
|
Boolean
|
|
fe_EditorCanPasteHtml(MWContext* context, int32* r_length)
|
|
{
|
|
Widget mainw = CONTEXT_WIDGET(context);
|
|
Display* display = XtDisplay(mainw);
|
|
Window window = XtWindow(mainw);
|
|
int cb_result;
|
|
unsigned long length;
|
|
|
|
cb_result = XmClipboardInquireLength(display, window,
|
|
XFE_CCP_HTML, &length);
|
|
|
|
if (cb_result == ClipboardSuccess && length > 0) {
|
|
*r_length = length;
|
|
return TRUE;
|
|
} else {
|
|
*r_length = 0;
|
|
return FALSE;
|
|
}
|
|
}
|
|
|
|
Boolean
|
|
fe_EditorCanPasteText(MWContext* context, int32* r_length)
|
|
{
|
|
Widget mainw = CONTEXT_WIDGET(context);
|
|
Display* display = XtDisplay(mainw);
|
|
Window window = XtWindow(mainw);
|
|
int cb_result;
|
|
unsigned long length;
|
|
|
|
cb_result = XmClipboardInquireLength(display, window,
|
|
XFE_CCP_TEXT, &length);
|
|
|
|
if (cb_result == ClipboardSuccess && length > 0) {
|
|
*r_length = length;
|
|
return TRUE;
|
|
} else {
|
|
*r_length = 0;
|
|
return FALSE;
|
|
}
|
|
}
|
|
|
|
Boolean
|
|
fe_EditorCanPasteLocal(MWContext* context, int32* r_length)
|
|
{
|
|
*r_length = 0;
|
|
return FALSE;
|
|
}
|
|
|
|
#ifdef OSF1
|
|
extern char *fe_GetInternalHtml();
|
|
extern char *fe_GetInternalText();
|
|
#endif
|
|
|
|
Boolean
|
|
fe_EditorCanPasteCache(MWContext* context, int32* r_length)
|
|
{
|
|
unsigned long text_len = 0;
|
|
unsigned long html_len = 0;
|
|
unsigned long data_len = 0;
|
|
XP_Bool sel_p = False;
|
|
|
|
/*
|
|
* We have an "new" unified internal clipboard....
|
|
* Check it.
|
|
*/
|
|
sel_p = fe_InternalSelection(True, &text_len, &html_len, &data_len);
|
|
|
|
if (sel_p) {
|
|
if (data_len > 0) {
|
|
*r_length = data_len;
|
|
return TRUE;
|
|
}
|
|
else {
|
|
if (html_len > 0) {
|
|
*r_length = html_len;
|
|
return TRUE;
|
|
}
|
|
else {
|
|
if (text_len > 0) {
|
|
*r_length = text_len;
|
|
return TRUE;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
*r_length = 0;
|
|
return FALSE;
|
|
}
|
|
|
|
Boolean
|
|
fe_EditorCanPaste(MWContext* context)
|
|
{
|
|
int32 len;
|
|
|
|
/*
|
|
* We have to check if we can do a local paste first,
|
|
* because we can hang the server if we try to re-own
|
|
* the selection. This is all very hokey. I need to
|
|
* talk to jamie.
|
|
*/
|
|
if (
|
|
fe_EditorCanPasteLocal(context, &len)
|
|
||
|
|
fe_EditorCanPasteCache(context, &len)
|
|
||
|
|
fe_EditorCanPasteData(context, &len)
|
|
||
|
|
fe_EditorCanPasteHtml(context, &len)
|
|
||
|
|
fe_EditorCanPasteText(context, &len))
|
|
{
|
|
return TRUE;
|
|
} else {
|
|
return FALSE;
|
|
}
|
|
}
|
|
|
|
void
|
|
fe_EditorCopyToClipboard(MWContext* context, char* string, Time timestamp)
|
|
{
|
|
fe_OwnEDTSelection (context, timestamp, True, string, NULL, NULL, 0);
|
|
}
|
|
|
|
static void
|
|
fe_editor_report_copy_error(MWContext* context, EDT_ClipboardResult code)
|
|
{
|
|
char* msg;
|
|
int id;
|
|
|
|
switch (code) {
|
|
case EDT_COP_DOCUMENT_BUSY: /* IDS_DOCUMENT_BUSY */
|
|
id = XFE_EDITOR_COPY_DOCUMENT_BUSY;
|
|
/* Cannot copy or cut at this time, try again later. */
|
|
break;
|
|
case EDT_COP_SELECTION_EMPTY: /* IDS_SELECTION_EMPTY */
|
|
id = XFE_EDITOR_COPY_SELECTION_EMPTY;
|
|
/* Nothing is selected. */
|
|
break;
|
|
case EDT_COP_SELECTION_CROSSES_TABLE_DATA_CELL:
|
|
id = XFE_EDITOR_COPY_SELECTION_CROSSES_TABLE_DATA_CELL;
|
|
/* The selection includes a table cell boundary. */
|
|
/* Deleting and copying are not allowed. */
|
|
break;
|
|
default:
|
|
return;
|
|
}
|
|
|
|
msg = XP_GetString(id);
|
|
FE_Alert(context, msg);
|
|
}
|
|
|
|
void
|
|
fe_EditorReportCopyError(MWContext* context, EDT_ClipboardResult code)
|
|
{
|
|
fe_editor_report_copy_error(context, code);
|
|
}
|
|
|
|
Boolean
|
|
fe_EditorOwnSelection(MWContext *context, Time time,
|
|
Boolean clip_p, Boolean cut_p)
|
|
{
|
|
char* text;
|
|
int32 text_len;
|
|
char* data;
|
|
int32 data_len;
|
|
EDT_ClipboardResult cut_result;
|
|
|
|
if (cut_p && clip_p) {
|
|
if (!fe_EditorCanCut(context)) {
|
|
#ifdef DEBUG_rhess
|
|
fprintf(stderr, "fe_EditorOwnSelection::[ can't CUT ]\n");
|
|
#endif
|
|
return False;
|
|
}
|
|
|
|
cut_result = EDT_CutSelection(context,
|
|
&text, &text_len,
|
|
&data, &data_len);
|
|
}
|
|
else {
|
|
if (!fe_EditorCanCopy(context)) {
|
|
#ifdef DEBUG_rhess
|
|
fprintf(stderr, "fe_EditorOwnSelection::[ can't COPY ]\n");
|
|
#endif
|
|
return False;
|
|
}
|
|
|
|
cut_result = EDT_CopySelection(context,
|
|
&text, &text_len,
|
|
&data, &data_len);
|
|
}
|
|
|
|
if (cut_result != EDT_COP_OK) {
|
|
fe_editor_report_copy_error(context, cut_result);
|
|
return False;
|
|
}
|
|
|
|
if ((!text || text_len <= 0) && (!data || data_len <= 0))
|
|
return False;
|
|
|
|
fe_OwnEDTSelection (context, time, clip_p, text, NULL, data, data_len);
|
|
|
|
return True;
|
|
}
|
|
|
|
void
|
|
fe_EditorDisownSelection(MWContext *context, Time time, Boolean clip_p)
|
|
{
|
|
Widget mainw = CONTEXT_WIDGET(context);
|
|
|
|
if (clip_p)
|
|
{
|
|
fe_DisownClipboard(mainw, time);
|
|
}
|
|
else
|
|
{
|
|
fe_DisownPrimarySelection(mainw, time);
|
|
|
|
#ifdef NOT_rhess
|
|
if (!fe_ContextHasPopups(context))
|
|
EDT_ClearSelection(context);
|
|
#endif
|
|
}
|
|
}
|
|
|
|
Boolean
|
|
fe_EditorCut(MWContext* context, Time timestamp)
|
|
{
|
|
Boolean own_p = False;
|
|
|
|
own_p = fe_EditorOwnSelection(context, timestamp, True, True);
|
|
|
|
fe_EditorUpdateToolbar(context, 0);
|
|
|
|
return own_p;
|
|
}
|
|
|
|
Boolean
|
|
fe_EditorCopy(MWContext* context, Time timestamp)
|
|
{
|
|
Boolean own_p = False;
|
|
|
|
own_p = fe_EditorOwnSelection(context, timestamp, True, False);
|
|
|
|
fe_EditorUpdateToolbar(context, 0);
|
|
|
|
return own_p;
|
|
}
|
|
|
|
static char fe_editor_local_name[] = "LOCAL";
|
|
|
|
#define XFE_CCP_LOCAL (fe_editor_local_name)
|
|
|
|
typedef struct {
|
|
MWContext* context;
|
|
char* text;
|
|
char* html;
|
|
char* data;
|
|
unsigned long text_len;
|
|
unsigned long html_len;
|
|
unsigned long data_len;
|
|
int count;
|
|
int max_count;
|
|
} xfe_pdata;
|
|
|
|
static xfe_pdata*
|
|
fe_new_pdata(MWContext* context, int max_count)
|
|
{
|
|
xfe_pdata* pData = (xfe_pdata*) XP_ALLOC(sizeof(xfe_pdata));
|
|
|
|
pData->context = context;
|
|
pData->text = NULL;
|
|
pData->html = NULL;
|
|
pData->data = NULL;
|
|
pData->text_len = 0;
|
|
pData->html_len = 0;
|
|
pData->data_len = 0;
|
|
pData->count = 0;
|
|
pData->max_count = max_count;
|
|
|
|
return( pData );
|
|
}
|
|
|
|
static void
|
|
fe_free_pdata(xfe_pdata* pData)
|
|
{
|
|
if (pData->text) {
|
|
XtFree(pData->text);
|
|
}
|
|
|
|
if (pData->html) {
|
|
XtFree(pData->html);
|
|
}
|
|
|
|
if (pData->data) {
|
|
XtFree(pData->data);
|
|
}
|
|
|
|
pData->context = NULL;
|
|
pData->text = NULL;
|
|
pData->html = NULL;
|
|
pData->data = NULL;
|
|
pData->text_len = 0;
|
|
pData->html_len = 0;
|
|
pData->data_len = 0;
|
|
pData->count = 0;
|
|
pData->max_count = 0;
|
|
|
|
XP_FREE(pData);
|
|
}
|
|
|
|
static void
|
|
fe_primary_sel_cb(Widget widget, XtPointer cdata, Atom *sel, Atom *type,
|
|
XtPointer value, unsigned long *len, int *format)
|
|
{
|
|
xfe_pdata* pData = (xfe_pdata*) cdata;
|
|
|
|
if (value) {
|
|
char* clip = (char *) value;
|
|
|
|
if (*type == XFE_DATA_ATOM) {
|
|
/*
|
|
* WARNING... [ this is actually HTML data, not HTML text ]
|
|
*/
|
|
#ifdef DEBUG_rhess
|
|
fprintf(stderr,
|
|
"fe_primary_sel_cb::[ data ][ %d ]\n", *len);
|
|
#endif
|
|
pData->count++;
|
|
pData->data_len = *len;
|
|
pData->data = clip;
|
|
}
|
|
else {
|
|
if (*type == XFE_HTML_ATOM) {
|
|
/*
|
|
* NOTE... [ this is HTML source text ]
|
|
*/
|
|
#ifdef DEBUG_rhess
|
|
fprintf(stderr,
|
|
"fe_primary_sel_cb::[ html ][ %d ]\n", *len);
|
|
#endif
|
|
pData->count++;
|
|
pData->html_len = *len;
|
|
pData->html = clip;
|
|
}
|
|
else {
|
|
if (*type == XFE_TEXT_ATOM) {
|
|
#ifdef DEBUG_rhess
|
|
fprintf(stderr,
|
|
"fe_primary_sel_cb::[ text ][ %d ]\n", *len);
|
|
#endif
|
|
pData->count++;
|
|
pData->text_len = *len;
|
|
pData->text = clip;
|
|
}
|
|
else {
|
|
#ifdef DEBUG_rhess
|
|
fprintf(stderr, "WARNING... [ unknown type atom ]\n");
|
|
#endif
|
|
}
|
|
}
|
|
}
|
|
}
|
|
else {
|
|
pData->count++;
|
|
|
|
#ifdef DEBUG_rhess
|
|
if (*type == XFE_DATA_ATOM) {
|
|
fprintf(stderr, "fe_primary_sel_cb::[ NULL ] - DATA\n");
|
|
}
|
|
else {
|
|
if (*type == XFE_HTML_ATOM) {
|
|
fprintf(stderr, "fe_primary_sel_cb::[ NULL ] - HTML\n");
|
|
}
|
|
else {
|
|
if (*type == XFE_TEXT_ATOM) {
|
|
fprintf(stderr, "fe_primary_sel_cb::[ NULL ] - TEXT\n");
|
|
}
|
|
else {
|
|
fprintf(stderr, "fe_primary_sel_cb::[ ERROR ]\n");
|
|
}
|
|
}
|
|
}
|
|
#endif
|
|
}
|
|
|
|
if (pData->count == pData->max_count) {
|
|
if (pData->data_len > 0) {
|
|
#ifdef DEBUG_rhess
|
|
fprintf(stderr,
|
|
"fe_primary_sel_cb::[ PasteHTML ][ %d ]\n",
|
|
pData->data_len);
|
|
#endif
|
|
EDT_PasteHTML(pData->context, pData->data,
|
|
ED_PASTE_NORMAL);
|
|
}
|
|
else {
|
|
if (pData->html_len > 0) {
|
|
#ifdef DEBUG_rhess
|
|
fprintf(stderr,
|
|
"fe_primary_sel_cb::[ PasteQuote ][ %d ]\n",
|
|
pData->html_len);
|
|
#endif
|
|
EDT_PasteQuoteBegin(pData->context, True);
|
|
EDT_PasteQuote(pData->context, pData->html);
|
|
EDT_PasteQuoteEnd(pData->context);
|
|
}
|
|
else {
|
|
if (pData->text_len > 0) {
|
|
#ifdef DEBUG_rhess
|
|
fprintf(stderr,
|
|
"fe_primary_sel_cb::[ PasteText ][ %d ]\n",
|
|
pData->text_len);
|
|
#endif
|
|
EDT_PasteText(pData->context, pData->text);
|
|
}
|
|
#ifdef DEBUG_rhess
|
|
else {
|
|
fprintf(stderr, "fe_primary_sel_cb::[ EMPTY ]\n");
|
|
}
|
|
#endif
|
|
}
|
|
}
|
|
fe_free_pdata(pData);
|
|
}
|
|
}
|
|
|
|
Boolean
|
|
fe_EditorPastePrimarySel(MWContext* context, Time timestamp)
|
|
{
|
|
Widget mainw = CONTEXT_WIDGET(context);
|
|
Display* dpy = XtDisplay(mainw);
|
|
char* data = NULL;
|
|
unsigned long len = 0;
|
|
unsigned long text_len = 0;
|
|
unsigned long html_len = 0;
|
|
unsigned long data_len = 0;
|
|
XP_Bool sel_p = False;
|
|
|
|
/*
|
|
* NOTE: this only works if we are talking to one server... [ oops ]
|
|
*/
|
|
if (XFE_TEXT_ATOM == 0)
|
|
XFE_TEXT_ATOM = XmInternAtom(dpy, XFE_CCP_TEXT, False);
|
|
|
|
if (XFE_HTML_ATOM == 0)
|
|
XFE_HTML_ATOM = XmInternAtom(dpy, XFE_CCP_HTML, False);
|
|
|
|
if (XFE_DATA_ATOM == 0)
|
|
XFE_DATA_ATOM = XmInternAtom(dpy, XFE_CCP_DATA, False);
|
|
|
|
/*
|
|
* NOTE: first, let's see if we have an internal selection...
|
|
*
|
|
*/
|
|
sel_p = fe_InternalSelection(False, &text_len, &html_len, &data_len);
|
|
|
|
if (sel_p) {
|
|
if (data_len > 0) {
|
|
data = fe_GetPrimaryData(&len);
|
|
|
|
#ifdef DEBUG_rhess
|
|
fprintf(stderr, "fe_EditorPastePrimarySel::[ DATA ]\n");
|
|
#endif
|
|
EDT_PasteHTML(context, data, ED_PASTE_NORMAL);
|
|
}
|
|
else {
|
|
if (html_len > 0) {
|
|
data = fe_GetPrimaryHtml();
|
|
|
|
#ifdef DEBUG_rhess
|
|
fprintf(stderr, "fe_EditorPastePrimarySel::[ HTML ]\n");
|
|
#endif
|
|
EDT_PasteQuoteBegin( context, True);
|
|
EDT_PasteQuote( context, data);
|
|
EDT_PasteQuoteEnd( context );
|
|
}
|
|
else {
|
|
if (text_len > 0) {
|
|
data = fe_GetPrimaryText();
|
|
|
|
#ifdef DEBUG_rhess
|
|
fprintf(stderr, "fe_EditorPastePrimarySel::[ TEXT ]\n");
|
|
#endif
|
|
EDT_PasteText(context, data);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
else {
|
|
xfe_pdata *pData = fe_new_pdata(context, 3);
|
|
Atom atoms[3];
|
|
XtPointer tags[3];
|
|
|
|
atoms[0] = XFE_TEXT_ATOM;
|
|
atoms[1] = XFE_HTML_ATOM;
|
|
atoms[2] = XFE_DATA_ATOM;
|
|
|
|
tags[0] = (XtPointer) pData;
|
|
tags[1] = (XtPointer) pData;
|
|
tags[2] = (XtPointer) pData;
|
|
|
|
XtGetSelectionValues(mainw,
|
|
XA_PRIMARY, atoms, 3,
|
|
fe_primary_sel_cb,
|
|
tags,
|
|
timestamp);
|
|
}
|
|
return False; /* keep compiler happy for now */
|
|
}
|
|
|
|
Boolean
|
|
fe_EditorPaste(MWContext* context, Time timestamp)
|
|
{
|
|
int32 length;
|
|
char* tmp_data = NULL;
|
|
char* clip_data = NULL;
|
|
char* clip_name = NULL; /* none */
|
|
int cb_result;
|
|
Boolean rv = FALSE;
|
|
Widget mainw = CONTEXT_WIDGET(context);
|
|
Display* display = XtDisplay(mainw);
|
|
Window window = XtWindow(mainw);
|
|
unsigned long len = 0;
|
|
unsigned long text_len = 0;
|
|
unsigned long html_len = 0;
|
|
unsigned long data_len = 0;
|
|
int i;
|
|
|
|
|
|
if (!clip_data) {
|
|
XP_Bool sel_p = fe_InternalSelection(True,
|
|
&text_len, &html_len, &data_len);
|
|
|
|
if (sel_p) {
|
|
if (data_len > 0) {
|
|
tmp_data = fe_GetInternalData(&len);
|
|
|
|
clip_data = XP_ALLOC(len+1);
|
|
memcpy(clip_data, tmp_data, len);
|
|
clip_name = XFE_CCP_DATA;
|
|
}
|
|
else {
|
|
if (html_len > 0) {
|
|
tmp_data = fe_GetInternalHtml();
|
|
|
|
clip_data = XP_STRDUP(tmp_data);
|
|
clip_name = XFE_CCP_HTML;
|
|
}
|
|
else {
|
|
if (text_len > 0) {
|
|
tmp_data = fe_GetInternalText();
|
|
|
|
clip_data = XP_STRDUP(tmp_data);
|
|
clip_name = XFE_CCP_TEXT;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if (!clip_data) { /* go look on real CLIPBOARD */
|
|
|
|
i = 0;
|
|
do {
|
|
cb_result = XmClipboardLock(display, window);
|
|
#ifdef DEBUG_rhess
|
|
fprintf(stderr, "lockEDT::[ %d ]\n", i);
|
|
#endif
|
|
i++;
|
|
} while (cb_result == ClipboardLocked);
|
|
|
|
if (fe_EditorCanPasteHtml(context, &length)) {
|
|
clip_data = XP_ALLOC(length+1);
|
|
clip_name = XFE_CCP_HTML;
|
|
} else if (fe_EditorCanPasteText(context, &length)) {
|
|
clip_data = XP_ALLOC(length+1);
|
|
clip_name = XFE_CCP_TEXT;
|
|
}
|
|
|
|
cb_result = XmClipboardUnlock(display, window, TRUE);
|
|
#ifdef DEBUG_rhess
|
|
fprintf(stderr, "fe_EditorPaste::[ unlock ]\n");
|
|
#endif
|
|
|
|
if (clip_data) {
|
|
|
|
#ifdef DEBUG_rhess
|
|
fprintf(stderr, "fe_EditorPaste::[ retrieve ]\n");
|
|
#endif
|
|
cb_result = xfe_clipboard_retrieve_work(mainw,
|
|
clip_name,
|
|
&clip_data,
|
|
&len,
|
|
timestamp);
|
|
#ifdef DEBUG_rhess
|
|
fprintf(stderr, "fe_EditorPaste::[ gotcha ]\n");
|
|
#endif
|
|
|
|
if (cb_result == ClipboardSuccess) {
|
|
clip_data[len] = '\0';
|
|
} else {
|
|
XP_FREE(clip_data);
|
|
clip_data = NULL;
|
|
clip_name = NULL;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (clip_name == XFE_CCP_DATA) {
|
|
#ifdef DEBUG_rhess
|
|
fprintf (stderr, "fe_EditorPaste::[ XFE_CCP_DATA ]\n");
|
|
#endif
|
|
rv = (EDT_PasteHTML(context, clip_data, ED_PASTE_NORMAL)
|
|
== EDT_COP_OK);
|
|
XP_FREE(clip_data);
|
|
} else if (clip_name == XFE_CCP_HTML) {
|
|
#ifdef DEBUG_rhess
|
|
fprintf (stderr, "fe_EditorPaste::[ XFE_CCP_HTML ]\n");
|
|
#endif
|
|
EDT_PasteQuoteBegin( context, True);
|
|
EDT_PasteQuote( context, clip_data);
|
|
EDT_PasteQuoteEnd( context );
|
|
XP_FREE(clip_data);
|
|
} else if (clip_name == XFE_CCP_TEXT) {
|
|
#ifdef DEBUG_rhess
|
|
fprintf (stderr, "fe_EditorPaste::[ XFE_CCP_TEXT ]\n");
|
|
#endif
|
|
rv = (EDT_PasteText(context, clip_data) == EDT_COP_OK);
|
|
XP_FREE(clip_data);
|
|
} else if (clip_name == XFE_CCP_LOCAL) {
|
|
#ifdef DEBUG_rhess
|
|
fprintf (stderr, "fe_EditorPaste::[ XFE_CCP_LOCAL ]\n");
|
|
#endif
|
|
rv = (EDT_PasteText(context, clip_data) == EDT_COP_OK);
|
|
} else {
|
|
fe_Bell(context);
|
|
}
|
|
|
|
#ifdef DONT_rhess
|
|
fe_EditorUpdateToolbar(context, 0);
|
|
#endif
|
|
return rv;
|
|
}
|
|
|
|
void
|
|
fe_EditorSelectAll(MWContext* context)
|
|
{
|
|
Display *dpy = XtDisplay(CONTEXT_WIDGET(context));
|
|
Time time = XtLastTimestampProcessed(dpy);
|
|
|
|
EDT_SelectAll(context);
|
|
|
|
fe_EditorOwnSelection (context, time, False, False);
|
|
fe_EditorUpdateToolbar(context, 0);
|
|
}
|
|
|
|
void
|
|
fe_EditorDeleteItem(MWContext* context, Boolean previous)
|
|
{
|
|
EDT_ClipboardResult result;
|
|
|
|
if (previous)
|
|
result = EDT_DeletePreviousChar(context);
|
|
else
|
|
result = EDT_DeleteChar(context);
|
|
|
|
if (result != EDT_COP_OK)
|
|
{
|
|
static Time lastErrTime = 0;
|
|
Time time = XtLastTimestampProcessed(XtDisplay(CONTEXT_WIDGET(context)));
|
|
unsigned int delta = (time - lastErrTime);
|
|
|
|
if (delta > XtGetMultiClickTime(XtDisplay(CONTEXT_WIDGET(context))))
|
|
fe_editor_report_copy_error(context, result);
|
|
lastErrTime = time;
|
|
}
|
|
|
|
fe_EditorUpdateToolbar(context, 0);
|
|
}
|
|
|
|
Boolean
|
|
fe_EditorHorizontalRulePropertiesCanDo(MWContext* context)
|
|
{
|
|
ED_ElementType type = EDT_GetCurrentElementType(context);
|
|
|
|
if (type == ED_ELEMENT_HRULE)
|
|
return TRUE;
|
|
else
|
|
return FALSE;
|
|
}
|
|
|
|
void
|
|
fe_EditorHorizontalRulePropertiesSet(MWContext* context, EDT_HorizRuleData* data)
|
|
{
|
|
if (fe_EditorHorizontalRulePropertiesCanDo(context)) {
|
|
EDT_SetHorizRuleData(context, data);
|
|
} else {
|
|
EDT_InsertHorizRule(context, data);
|
|
}
|
|
}
|
|
|
|
void
|
|
fe_EditorHorizontalRulePropertiesGet(MWContext* context, EDT_HorizRuleData* data)
|
|
{
|
|
EDT_HorizRuleData* foo;
|
|
|
|
if (fe_EditorHorizontalRulePropertiesCanDo(context)
|
|
&&
|
|
(foo = EDT_GetHorizRuleData(context)) != NULL) {
|
|
memcpy(data, foo, sizeof(EDT_HorizRuleData));
|
|
EDT_FreeHorizRuleData(foo);
|
|
} else {
|
|
data->size = 2;
|
|
data->iWidth = 100;
|
|
data->bWidthPercent = TRUE;
|
|
data->align = ED_ALIGN_CENTER;
|
|
data->bNoShade = FALSE;
|
|
}
|
|
}
|
|
|
|
void
|
|
fe_EditorRefresh(MWContext* context)
|
|
{
|
|
EDT_RefreshLayout(context);
|
|
}
|
|
|
|
void
|
|
fe_EditorHrefInsert(MWContext* context, char* p_text, char* p_url)
|
|
{
|
|
char* text;
|
|
char* url;
|
|
EDT_ClipboardResult result;
|
|
|
|
if (!EDT_CanSetHREF(context)) { /* should not exist already */
|
|
|
|
if (p_url && p_url[0] != '\0')
|
|
url = XP_STRDUP(p_url);
|
|
else
|
|
return; /* no url specified */
|
|
|
|
if (p_text && p_text[0] != '\0')
|
|
text = XP_STRDUP(p_text);
|
|
else
|
|
text = XP_STRDUP(p_url);
|
|
|
|
result = EDT_PasteHREF(context, &url, &text, 1);
|
|
|
|
if (result != EDT_COP_OK)
|
|
fe_editor_report_copy_error(context, result);
|
|
}
|
|
}
|
|
|
|
Boolean
|
|
fe_EditorGetHref(MWContext *context) {
|
|
char* p;
|
|
|
|
p = (char *) LO_GetSelectionText(context);
|
|
if (p == NULL) return False;
|
|
else
|
|
return NULL != EDT_GetHREF(context);
|
|
}
|
|
|
|
|
|
Boolean
|
|
fe_EditorHrefGet(MWContext* context, char** text, char** url)
|
|
{
|
|
EDT_HREFData* href_data = EDT_GetHREFData(context);
|
|
Bool is_selected = EDT_IsSelected(context);
|
|
char* p;
|
|
|
|
p = (char *) LO_GetSelectionText(context);
|
|
|
|
if (is_selected && p != NULL) {
|
|
*text = XtNewString(p);
|
|
XP_FREE(p);
|
|
} else if (p == NULL) { /* this fixes #27762 */
|
|
*text = NULL;
|
|
} else if ((p = EDT_GetHREFText(context)) != NULL) {
|
|
/* charlie's new get the text of a link thing */
|
|
*text = p;
|
|
} else {
|
|
*text = NULL;
|
|
}
|
|
|
|
if (href_data != NULL && (p = href_data->pURL) != NULL) {
|
|
*url = XtNewString(p);
|
|
} else {
|
|
*url = NULL;
|
|
}
|
|
|
|
if (href_data != NULL)
|
|
EDT_FreeHREFData(href_data);
|
|
|
|
return EDT_CanSetHREF(context);
|
|
}
|
|
|
|
void
|
|
fe_EditorHrefSetUrl(MWContext* context, char* url)
|
|
{
|
|
EDT_HREFData* href_data;
|
|
|
|
if (EDT_CanSetHREF(context)) {
|
|
|
|
href_data = EDT_GetHREFData(context);
|
|
|
|
if (!href_data)
|
|
href_data = EDT_NewHREFData();
|
|
|
|
if (href_data->pURL)
|
|
XP_FREE(href_data->pURL);
|
|
|
|
if (url)
|
|
href_data->pURL = XP_STRDUP(url);
|
|
else
|
|
href_data->pURL = NULL; /* clear the href */
|
|
|
|
EDT_SetHREFData(context, href_data);
|
|
|
|
EDT_FreeHREFData(href_data);
|
|
}
|
|
}
|
|
|
|
void
|
|
fe_EditorHrefClearUrl(MWContext* context)
|
|
{
|
|
fe_EditorHrefSetUrl(context, NULL);
|
|
}
|
|
|
|
void
|
|
fe_EditorRemoveLinks(MWContext* context)
|
|
{
|
|
/*
|
|
* Remove a single link or all links within selected region.
|
|
*/
|
|
if (EDT_CanSetHREF(context)) {
|
|
EDT_SetHREF(context, NULL);
|
|
}
|
|
}
|
|
|
|
static Time
|
|
fe_getTimeFromEvent(MWContext* context, XEvent* event)
|
|
{
|
|
Time time;
|
|
time = (event && (event->type == KeyPress ||
|
|
event->type == KeyRelease)
|
|
? event->xkey.time :
|
|
event && (event->type == ButtonPress ||
|
|
event->type == ButtonRelease)
|
|
? event->xbutton.time :
|
|
XtLastTimestampProcessed (XtDisplay(CONTEXT_WIDGET (context))));
|
|
return time;
|
|
}
|
|
|
|
Boolean
|
|
fe_EditorPropertiesDialogCanDo(MWContext* context,
|
|
fe_EditorPropertiesDialogType d_type)
|
|
{
|
|
ED_ElementType e_type;
|
|
|
|
if (
|
|
d_type == XFE_EDITOR_PROPERTIES_IMAGE_INSERT
|
|
||
|
|
d_type == XFE_EDITOR_PROPERTIES_LINK_INSERT
|
|
||
|
|
d_type == XFE_EDITOR_PROPERTIES_CHARACTER
|
|
||
|
|
d_type == XFE_EDITOR_PROPERTIES_PARAGRAPH
|
|
||
|
|
d_type == XFE_EDITOR_PROPERTIES_TABLE
|
|
||
|
|
d_type == XFE_EDITOR_PROPERTIES_DOCUMENT
|
|
)
|
|
return TRUE; /* always ok to do */
|
|
|
|
e_type = EDT_GetCurrentElementType(context);
|
|
|
|
if (d_type == XFE_EDITOR_PROPERTIES_IMAGE) {
|
|
if (e_type == ED_ELEMENT_IMAGE)
|
|
return TRUE;
|
|
else
|
|
return FALSE;
|
|
|
|
} else if (d_type == XFE_EDITOR_PROPERTIES_HRULE) {
|
|
if (e_type == ED_ELEMENT_HRULE)
|
|
return TRUE;
|
|
else
|
|
return FALSE;
|
|
|
|
} else if (d_type == XFE_EDITOR_PROPERTIES_HTML_TAG) {
|
|
if (e_type == ED_ELEMENT_UNKNOWN_TAG)
|
|
return TRUE;
|
|
else
|
|
return FALSE;
|
|
|
|
} else if (d_type == XFE_EDITOR_PROPERTIES_TARGET) {
|
|
if (e_type == ED_ELEMENT_TARGET)
|
|
return TRUE;
|
|
else
|
|
return FALSE;
|
|
|
|
} else if (d_type == XFE_EDITOR_PROPERTIES_LINK) {
|
|
if (EDT_GetHREF(context))
|
|
return TRUE;
|
|
else
|
|
return FALSE;
|
|
}
|
|
return FALSE;
|
|
}
|
|
|
|
/*
|
|
* This stuff is so crazy.
|
|
*
|
|
*/
|
|
void
|
|
fe_EditorParagraphPropertiesSetAll(MWContext* context,
|
|
TagType paragragh_type,
|
|
EDT_ListData* list_data,
|
|
ED_Alignment align)
|
|
{
|
|
TagType old_paragraph_type = EDT_GetParagraphFormatting(context);
|
|
EDT_ListData* old_list_data = EDT_GetListData(context);
|
|
Boolean do_set;
|
|
|
|
EDT_BeginBatchChanges(context);
|
|
|
|
/*
|
|
* Windows has this kludge, so...
|
|
*/
|
|
if (align == ED_ALIGN_LEFT)
|
|
align = ED_ALIGN_DEFAULT;
|
|
|
|
EDT_SetParagraphAlign(context, align);
|
|
|
|
/*
|
|
* If there was a list before, and now there is none,
|
|
* remove the list.
|
|
*/
|
|
if (old_list_data && !list_data) {
|
|
for ( ; old_list_data; old_list_data = EDT_GetListData(context)) {
|
|
EDT_FreeListData(old_list_data);
|
|
EDT_Outdent(context);
|
|
}
|
|
old_list_data = NULL;
|
|
}
|
|
|
|
/*
|
|
* If the style has changed.
|
|
*/
|
|
if (paragragh_type != old_paragraph_type)
|
|
EDT_MorphContainer(context, paragragh_type);
|
|
|
|
/*
|
|
* If there was no list, and now there is..
|
|
*/
|
|
if (!old_list_data && list_data) {
|
|
EDT_Indent(context);
|
|
old_list_data = EDT_GetListData(context);
|
|
}
|
|
|
|
/*
|
|
* The BE is losing. It cannot deal with lists in selections,
|
|
* it returns no list data. So, we protect oursleves and
|
|
* end up doing nothing, just like windows.
|
|
*/
|
|
if (old_list_data != NULL) {
|
|
|
|
do_set = TRUE;
|
|
|
|
/*
|
|
* If this is a block quote..
|
|
*/
|
|
if (list_data != NULL && list_data->iTagType == P_BLOCKQUOTE) {
|
|
old_list_data->iTagType = P_BLOCKQUOTE;
|
|
old_list_data->eType = ED_LIST_TYPE_DEFAULT;
|
|
}
|
|
|
|
/*
|
|
* If it's a list.
|
|
*/
|
|
else if (list_data != NULL) {
|
|
old_list_data->iTagType = list_data->iTagType;
|
|
old_list_data->eType = ED_LIST_TYPE_DEFAULT;
|
|
#if 0
|
|
/*
|
|
* We have no UI for this, but maybe it's in the author's
|
|
* HTML. Don't change it. Actually, I have no idea.
|
|
*/
|
|
old_list_data->bCompact = FALSE;
|
|
#endif
|
|
switch (list_data->iTagType) {
|
|
case P_UNUM_LIST:
|
|
if (list_data->eType == ED_LIST_TYPE_DISC ||
|
|
list_data->eType == ED_LIST_TYPE_SQUARE ||
|
|
list_data->eType == ED_LIST_TYPE_CIRCLE)
|
|
old_list_data->eType = list_data->eType;
|
|
break;
|
|
case P_NUM_LIST:
|
|
if (list_data->eType == ED_LIST_TYPE_DIGIT ||
|
|
list_data->eType == ED_LIST_TYPE_BIG_ROMAN ||
|
|
list_data->eType == ED_LIST_TYPE_SMALL_ROMAN ||
|
|
list_data->eType == ED_LIST_TYPE_BIG_LETTERS ||
|
|
list_data->eType == ED_LIST_TYPE_SMALL_LETTERS)
|
|
old_list_data->eType = list_data->eType;
|
|
if (list_data->iStart > 0)
|
|
old_list_data->iStart = list_data->iStart;
|
|
else
|
|
old_list_data->iStart = 1;
|
|
break;
|
|
case P_DIRECTORY:
|
|
case P_MENU:
|
|
case P_DESC_LIST:
|
|
break;
|
|
default: /* garbage, don't set */
|
|
do_set = FALSE;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (do_set)
|
|
EDT_SetListData(context, old_list_data);
|
|
}
|
|
|
|
/*
|
|
* Else it's just an ordinary old jo.
|
|
*/
|
|
/* nothing to do */
|
|
|
|
if (old_list_data != NULL)
|
|
EDT_FreeListData(old_list_data);
|
|
|
|
EDT_EndBatchChanges(context);
|
|
|
|
fe_EditorUpdateToolbar(context, 0);
|
|
}
|
|
|
|
Boolean
|
|
fe_EditorParagraphPropertiesGetAll(MWContext* context,
|
|
TagType* paragragh_type,
|
|
EDT_ListData* list_data,
|
|
ED_Alignment* align)
|
|
{
|
|
TagType old_paragraph_type = EDT_GetParagraphFormatting(context);
|
|
EDT_ListData* old_list_data = EDT_GetListData(context);
|
|
|
|
if (paragragh_type)
|
|
*paragragh_type = old_paragraph_type;
|
|
|
|
if (list_data && old_list_data)
|
|
*list_data = *old_list_data;
|
|
|
|
if (align)
|
|
*align = EDT_GetParagraphAlign(context);
|
|
|
|
if (old_list_data != NULL)
|
|
EDT_FreeListData(old_list_data);
|
|
|
|
return (old_list_data != NULL);
|
|
}
|
|
|
|
/*FIXME*/
|
|
/*
|
|
* I think the following routine could now be replaced
|
|
* by EDT_SetFontColor(context, &color);
|
|
*/
|
|
void
|
|
fe_EditorColorSet(MWContext* context, LO_Color* color)
|
|
{
|
|
EDT_CharacterData cdata;
|
|
|
|
memset(&cdata, 0, sizeof(EDT_CharacterData));
|
|
|
|
cdata.mask = TF_FONT_COLOR;
|
|
cdata.pColor = color;
|
|
|
|
if (color)
|
|
cdata.values = TF_FONT_COLOR;
|
|
else
|
|
cdata.values = 0;
|
|
|
|
EDT_SetCharacterData(context, &cdata);
|
|
}
|
|
|
|
Boolean
|
|
fe_EditorColorGet(MWContext* context, LO_Color* color)
|
|
{
|
|
EDT_CharacterData* cdata = EDT_GetCharacterData(context);
|
|
Boolean rv = FALSE;
|
|
LO_Color* use_color = NULL;
|
|
|
|
if (cdata != NULL && (cdata->values & TF_FONT_COLOR) != 0) { /* custom */
|
|
rv = TRUE; /* custom */
|
|
if ((cdata->mask & TF_FONT_COLOR) != 0 && cdata->pColor != NULL) {
|
|
use_color = cdata->pColor; /* defined */
|
|
}
|
|
}
|
|
|
|
if (use_color != NULL) {
|
|
*color = *use_color;
|
|
} else {
|
|
fe_EditorDocumentGetColors(context,
|
|
NULL, /* bg image */
|
|
NULL, /* leave image */
|
|
NULL, /* bg color */
|
|
color,/* normal color */
|
|
NULL, /* link color */
|
|
NULL, /* active color */
|
|
NULL); /* followed color */
|
|
}
|
|
|
|
if (cdata != NULL)
|
|
EDT_FreeCharacterData(cdata);
|
|
|
|
return rv; /* custom */
|
|
}
|
|
|
|
extern int XFE_COULDNT_FORK;
|
|
extern int XFE_EXECVP_FAILED;
|
|
|
|
void
|
|
fe_ExecCommand(MWContext* context, char** argv)
|
|
{
|
|
char buf[MAXPATHLEN];
|
|
int forked;
|
|
|
|
switch (forked = fork()) {
|
|
case -1:
|
|
fe_perror(context, XP_GetString(XFE_COULDNT_FORK));
|
|
break;
|
|
case 0: {
|
|
Display *dpy = XtDisplay(CONTEXT_WIDGET(context));
|
|
close(ConnectionNumber(dpy));
|
|
|
|
execvp(argv[0], argv);
|
|
|
|
PR_snprintf(buf, sizeof(buf), XP_GetString(XFE_EXECVP_FAILED),
|
|
fe_progname, argv[0]);
|
|
perror(buf);
|
|
exit(3); /* Note that this only exits a child fork. */
|
|
break;
|
|
}
|
|
default:
|
|
/* block */
|
|
break;
|
|
}
|
|
}
|
|
|
|
static char**
|
|
fe_editor_parse_argv(char* string, char* filename)
|
|
{
|
|
char buf[MAXPATHLEN];
|
|
unsigned size = 32;
|
|
unsigned argc = 0;
|
|
char** argv = (char**)malloc(sizeof(char*) * size);
|
|
char* p;
|
|
char* q;
|
|
char* r;
|
|
|
|
for (p = string; *p != '\0'; ) {
|
|
|
|
q = buf;
|
|
|
|
/* skip leading white */
|
|
while (isspace(*p))
|
|
p++;
|
|
|
|
while (!isspace(*p) && *p != '\0' && ((q - buf) < sizeof(buf))) {
|
|
if (*p == '%' && p[1] == 'f') {
|
|
for (r = filename; *r != '\0'; ) {
|
|
*q++ = *r++;
|
|
}
|
|
p += 2;
|
|
} else {
|
|
*q++ = *p++;
|
|
}
|
|
}
|
|
*q = '\0';
|
|
|
|
if (q > buf)
|
|
argv[argc++] = strdup(buf);
|
|
|
|
if (argc == size) {
|
|
size = argc + 32;
|
|
argv = (char**)realloc((void*)argv, sizeof(char*) * size);
|
|
}
|
|
}
|
|
|
|
if (argc > 0) {
|
|
argv[argc] = NULL;
|
|
return argv;
|
|
} else {
|
|
free(argv);
|
|
return NULL;
|
|
}
|
|
}
|
|
|
|
void
|
|
fe_editor_free_argv(char** argv)
|
|
{
|
|
unsigned i;
|
|
for (i = 0; argv[i]; i++)
|
|
free(argv[i]);
|
|
free(argv);
|
|
}
|
|
|
|
void
|
|
fe_editor_exec_command(MWContext* context, char* command, char* filename)
|
|
{
|
|
char** argv;
|
|
|
|
if ((argv = fe_editor_parse_argv(command, filename)) != NULL) {
|
|
|
|
fe_ExecCommand(context, argv);
|
|
|
|
fe_editor_free_argv(argv);
|
|
} else {
|
|
/* Empty command specified! */
|
|
FE_Alert(context, XP_GetString(XFE_COMMAND_EMPTY));
|
|
}
|
|
}
|
|
|
|
void
|
|
fe_EditorEditSource(MWContext* context)
|
|
{
|
|
char* html_editor;
|
|
char* local_name;
|
|
History_entry* hist_entry;
|
|
|
|
if (!FE_CheckAndSaveDocument(context))
|
|
return;
|
|
fe_EditorPreferencesGetEditors(context, &html_editor, NULL);
|
|
|
|
if (html_editor == NULL || XP_STRSTR(html_editor, "%f") == NULL) {
|
|
/*
|
|
* "Please specify an editor in Editor Preferences.\n"
|
|
* "Specify the file argument with %f. Netscape will\n"
|
|
* "replace %f with the correct file name. Examples:\n"
|
|
* " xterm -e vi %f\n"
|
|
* " xgifedit %f\n"
|
|
*/
|
|
char* msg = XP_GetString(XFE_EDITOR_HTML_EDITOR_COMMAND_EMPTY);
|
|
|
|
if (XFE_Confirm(context, msg)) {
|
|
fe_EditorShowNewPreferences(context); /* call C++ adaptor */
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
hist_entry = SHIST_GetCurrent(&context->hist);
|
|
if (hist_entry && hist_entry->address) {
|
|
if (XP_ConvertUrlToLocalFile(hist_entry->address, &local_name)) {
|
|
fe_editor_exec_command(context, html_editor, local_name);
|
|
XP_FREE(local_name);
|
|
}
|
|
}
|
|
}
|
|
|
|
void
|
|
fe_EditorEditImage(MWContext* context, char* file)
|
|
{
|
|
char* image_editor;
|
|
char* local_name = NULL;
|
|
char* full_file;
|
|
char* directory;
|
|
char* p;
|
|
History_entry* hist_entry;
|
|
|
|
fe_EditorPreferencesGetEditors(context, NULL, &image_editor);
|
|
|
|
if (image_editor == NULL || XP_STRSTR(image_editor, "%f") == NULL) {
|
|
/*
|
|
* "Please specify an editor in Editor Preferences.\n"
|
|
* "Specify the file argument with %f. Netscape will\n"
|
|
* "replace %f with the correct file name. Examples:\n"
|
|
* " xterm -e vi %f\n"
|
|
* " xgifedit %f\n"
|
|
*/
|
|
char* msg = XP_GetString(XFE_EDITOR_IMAGE_EDITOR_COMMAND_EMPTY);
|
|
|
|
if (XFE_Confirm(context, msg)) {
|
|
fe_EditorShowNewPreferences(context); /* call C++ adaptor */
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
while (isspace(*file))
|
|
file++;
|
|
|
|
if (file[0] != '/') { /* not absolute */
|
|
hist_entry = SHIST_GetCurrent(&context->hist);
|
|
if (hist_entry && hist_entry->address) {
|
|
if (XP_ConvertUrlToLocalFile(hist_entry->address, &local_name)) {
|
|
p = strrchr(local_name, '/'); /* find last slash */
|
|
if (p) {
|
|
*p = '\0';
|
|
directory = local_name;
|
|
} else {
|
|
directory = ".";
|
|
}
|
|
full_file = (char*)XtMalloc(strlen(directory) +
|
|
strlen(file) + 2);
|
|
sprintf(full_file, "%s/%s", directory, file);
|
|
XP_FREE(local_name);
|
|
|
|
fe_editor_exec_command(context, image_editor, full_file);
|
|
XtFree(full_file);
|
|
}
|
|
}
|
|
} else {
|
|
fe_editor_exec_command(context, image_editor, file);
|
|
}
|
|
}
|
|
|
|
#define COPY_SAVE_EXTRA(b, a) \
|
|
{ char* xx_extra = (b)->pExtra; *(b) = *(a); (b)->pExtra = xx_extra; }
|
|
|
|
#define COPY_SAVE_BGCOLOR(b, a) \
|
|
{ char* xx_bgcolor = (b)->pColorBackground; *(b) = *(a); (b)->pColorBackground = xx_bgcolor; }
|
|
|
|
static LO_Color*
|
|
fe_dup_lo_color(LO_Color* color)
|
|
{
|
|
LO_Color* new_color;
|
|
|
|
if (color != NULL && (new_color = (LO_Color*)XtNew(LO_Color)) != NULL) {
|
|
*new_color = *color;
|
|
return new_color;
|
|
} else {
|
|
return NULL;
|
|
}
|
|
}
|
|
|
|
#define DUP_BGCOLOR(b, a) \
|
|
{ (b)->pColorBackground = fe_dup_lo_color((a)->pColorBackground); }
|
|
|
|
/*
|
|
* Table support.
|
|
*/
|
|
Boolean
|
|
fe_EditorTableCanInsert(MWContext* context)
|
|
{
|
|
return !EDT_IsJavaScript(context);
|
|
}
|
|
|
|
void
|
|
fe_EditorTableInsert(MWContext* context, EDT_AllTableData* all_data)
|
|
{
|
|
EDT_TableData* new_data = EDT_NewTableData();
|
|
EDT_TableCaptionData* caption_data;
|
|
|
|
if (all_data) {
|
|
EDT_TableData* data = &all_data->td;
|
|
COPY_SAVE_EXTRA(new_data, data);
|
|
DUP_BGCOLOR(new_data, data);
|
|
} else {
|
|
new_data->iBorderWidth = 1;
|
|
new_data->iRows = 2;
|
|
new_data->iColumns = 2;
|
|
}
|
|
EDT_BeginBatchChanges(context);
|
|
|
|
EDT_InsertTable(context, new_data);
|
|
|
|
if (all_data != NULL && all_data->has_caption) {
|
|
caption_data = EDT_NewTableCaptionData();
|
|
caption_data->align = all_data->cd.align;
|
|
EDT_InsertTableCaption(context, caption_data);
|
|
}
|
|
|
|
EDT_FreeTableData(new_data);
|
|
|
|
EDT_EndBatchChanges(context);
|
|
}
|
|
|
|
Boolean
|
|
fe_EditorTableCanDelete(MWContext* context)
|
|
{
|
|
return (EDT_IsInsertPointInTable(context) != 0);
|
|
}
|
|
|
|
void
|
|
fe_EditorTableDelete(MWContext* context)
|
|
{
|
|
if (EDT_IsInsertPointInTable(context) != 0) {
|
|
EDT_DeleteTable(context);
|
|
fe_EditorUpdateToolbar(context, 0);
|
|
}
|
|
}
|
|
|
|
Boolean
|
|
fe_EditorTableGetData(MWContext* context, EDT_AllTableData* all_data)
|
|
{
|
|
Boolean in_table = FALSE;
|
|
EDT_TableData* table_data = EDT_GetTableData(context);
|
|
EDT_TableData* pt_data = &all_data->td;
|
|
EDT_TableCaptionData* caption_data;
|
|
|
|
if (table_data) {
|
|
*pt_data = *table_data;
|
|
if (table_data->pColorBackground != NULL) {
|
|
static LO_Color color_buf;
|
|
color_buf = *table_data->pColorBackground;
|
|
pt_data->pColorBackground = &color_buf; /*caller cannot hang on*/
|
|
}
|
|
|
|
if (table_data->pBackgroundImage != NULL) {
|
|
static char* bg_image;
|
|
if (bg_image != NULL)
|
|
XP_FREE(bg_image);
|
|
bg_image = XP_STRDUP(table_data->pBackgroundImage);
|
|
pt_data->pBackgroundImage = bg_image; /*caller cannot hang on*/
|
|
}
|
|
|
|
EDT_FreeTableData(table_data);
|
|
in_table = TRUE;
|
|
} else {
|
|
return FALSE;
|
|
}
|
|
|
|
if (EDT_IsInsertPointInTableCaption(context)
|
|
&&
|
|
(caption_data = EDT_GetTableCaptionData(context)) != NULL) {
|
|
|
|
all_data->has_caption = TRUE;
|
|
all_data->cd.align = caption_data->align;
|
|
} else {
|
|
all_data->has_caption = FALSE;
|
|
all_data->cd.align = ED_ALIGN_TOP;
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
void
|
|
fe_EditorTableSetData(MWContext* context, EDT_AllTableData* all_data)
|
|
{
|
|
Boolean has_caption;
|
|
Boolean wants_caption;
|
|
EDT_TableData* table_data = EDT_GetTableData(context);
|
|
EDT_TableCaptionData* caption_data;
|
|
|
|
if (table_data) {
|
|
EDT_BeginBatchChanges(context);
|
|
|
|
/* get the basic table data */
|
|
COPY_SAVE_EXTRA(table_data, &all_data->td);
|
|
DUP_BGCOLOR(table_data, &all_data->td);
|
|
|
|
EDT_SetTableData(context, table_data);
|
|
|
|
caption_data = NULL;
|
|
has_caption = EDT_IsInsertPointInTableCaption(context);
|
|
wants_caption = all_data->has_caption;
|
|
|
|
if (has_caption == TRUE && wants_caption == FALSE) {
|
|
EDT_DeleteTableCaption(context);
|
|
} else if (has_caption == FALSE && wants_caption == TRUE) {
|
|
caption_data = EDT_NewTableCaptionData();
|
|
caption_data->align = all_data->cd.align;
|
|
EDT_InsertTableCaption(context, caption_data);
|
|
} else if (has_caption
|
|
&&
|
|
(caption_data = EDT_GetTableCaptionData(context)) != NULL) {
|
|
if (caption_data->align != all_data->cd.align) {
|
|
caption_data->align = all_data->cd.align;
|
|
EDT_SetTableCaptionData(context, caption_data);
|
|
}
|
|
}
|
|
|
|
EDT_BeginBatchChanges(context);
|
|
|
|
if (caption_data)
|
|
EDT_FreeTableCaptionData(caption_data);
|
|
|
|
EDT_FreeTableData(table_data);
|
|
}
|
|
}
|
|
|
|
Boolean
|
|
fe_EditorTableRowCanInsert(MWContext* context)
|
|
{
|
|
return (EDT_IsInsertPointInTableRow(context) != 0);
|
|
}
|
|
|
|
void
|
|
fe_EditorTableRowInsert(MWContext* context, EDT_TableRowData* data)
|
|
{
|
|
EDT_TableRowData* new_data;
|
|
|
|
if (!fe_EditorTableRowCanInsert(context)) {
|
|
fe_Bell(context);
|
|
return;
|
|
}
|
|
|
|
new_data = EDT_GetTableRowData(context);
|
|
|
|
if (!new_data)
|
|
return;
|
|
|
|
if (data) {
|
|
new_data->align = data->align;
|
|
new_data->valign = data->valign;
|
|
DUP_BGCOLOR(new_data, data);
|
|
}
|
|
EDT_InsertTableRows(context, new_data, TRUE, 1);
|
|
EDT_FreeTableRowData(new_data);
|
|
fe_EditorUpdateToolbar(context, 0);
|
|
}
|
|
|
|
Boolean
|
|
fe_EditorTableRowCanDelete(MWContext* context)
|
|
{
|
|
return (EDT_IsInsertPointInTableRow(context) != 0);
|
|
}
|
|
|
|
void
|
|
fe_EditorTableRowDelete(MWContext* context)
|
|
{
|
|
if (EDT_IsInsertPointInTable(context) != 0) {
|
|
EDT_DeleteTableRows(context, 1);
|
|
fe_EditorUpdateToolbar(context, 0);
|
|
}
|
|
}
|
|
|
|
Boolean
|
|
fe_EditorTableRowGetData(MWContext* context, EDT_TableRowData* p_data)
|
|
{
|
|
EDT_TableRowData* row_data = EDT_GetTableRowData(context);
|
|
|
|
if (row_data) {
|
|
*p_data = *row_data;
|
|
if (row_data->pColorBackground != NULL) {
|
|
static LO_Color color_buf;
|
|
color_buf = *row_data->pColorBackground;
|
|
p_data->pColorBackground = &color_buf; /*caller cannot hang on*/
|
|
}
|
|
EDT_FreeTableRowData(row_data);
|
|
return TRUE;
|
|
} else {
|
|
return FALSE;
|
|
}
|
|
}
|
|
|
|
void
|
|
fe_EditorTableRowSetData(MWContext* context, EDT_TableRowData* p_data)
|
|
{
|
|
EDT_TableRowData* row_data = EDT_GetTableRowData(context);
|
|
|
|
if (row_data) {
|
|
COPY_SAVE_EXTRA(row_data, p_data);
|
|
DUP_BGCOLOR(row_data, p_data);
|
|
EDT_SetTableRowData(context, row_data);
|
|
EDT_FreeTableRowData(row_data);
|
|
fe_EditorUpdateToolbar(context, 0);
|
|
}
|
|
}
|
|
|
|
Boolean
|
|
fe_EditorTableCaptionCanInsert(MWContext* context)
|
|
{
|
|
return (EDT_IsInsertPointInTable(context) != 0);
|
|
}
|
|
|
|
void
|
|
fe_EditorTableCaptionInsert(MWContext* context, EDT_TableCaptionData* data)
|
|
{
|
|
EDT_TableCaptionData* new_data = EDT_NewTableCaptionData();
|
|
EDT_InsertTableCaption(context, new_data);
|
|
EDT_FreeTableCaptionData(new_data);
|
|
fe_EditorUpdateToolbar(context, 0);
|
|
}
|
|
|
|
Boolean
|
|
fe_EditorTableCaptionCanDelete(MWContext* context)
|
|
{
|
|
return (EDT_IsInsertPointInTableCaption(context) != 0);
|
|
}
|
|
|
|
void
|
|
fe_EditorTableCaptionDelete(MWContext* context)
|
|
{
|
|
if (EDT_IsInsertPointInTableCaption(context) != 0) {
|
|
EDT_DeleteTableCaption(context);
|
|
fe_EditorUpdateToolbar(context, 0);
|
|
}
|
|
}
|
|
|
|
Boolean
|
|
fe_EditorTableColumnCanInsert(MWContext* context)
|
|
{
|
|
return (EDT_IsInsertPointInTableRow(context) != 0);
|
|
}
|
|
|
|
void
|
|
fe_EditorTableColumnInsert(MWContext* context, EDT_TableCellData* data)
|
|
{
|
|
EDT_TableCellData* new_data = EDT_NewTableCellData();
|
|
EDT_InsertTableColumns(context, new_data, TRUE, 1);
|
|
EDT_FreeTableCellData(new_data);
|
|
|
|
fe_EditorUpdateToolbar(context, 0);
|
|
}
|
|
|
|
Boolean
|
|
fe_EditorTableColumnCanDelete(MWContext* context)
|
|
{
|
|
return (EDT_IsInsertPointInTableRow(context) != 0);
|
|
}
|
|
|
|
void
|
|
fe_EditorTableColumnDelete(MWContext* context)
|
|
{
|
|
if (EDT_IsInsertPointInTableRow(context) != 0) {
|
|
EDT_DeleteTableColumns(context, 1);
|
|
fe_EditorUpdateToolbar(context, 0);
|
|
}
|
|
}
|
|
|
|
Boolean
|
|
fe_EditorTableCellCanInsert(MWContext* context)
|
|
{
|
|
return (EDT_IsInsertPointInTableRow(context) != 0);
|
|
}
|
|
|
|
void
|
|
fe_EditorTableCellInsert(MWContext* context, EDT_TableCellData* data)
|
|
{
|
|
EDT_TableCellData* new_data = EDT_NewTableCellData();
|
|
EDT_InsertTableCells(context, new_data, TRUE, 1);
|
|
EDT_FreeTableCellData(new_data);
|
|
|
|
fe_EditorUpdateToolbar(context, 0);
|
|
}
|
|
|
|
Boolean
|
|
fe_EditorTableCellCanDelete(MWContext* context)
|
|
{
|
|
return (EDT_IsInsertPointInTableCell(context) != 0);
|
|
}
|
|
|
|
void
|
|
fe_EditorTableCellDelete(MWContext* context)
|
|
{
|
|
if (EDT_IsInsertPointInTableRow(context) != 0) {
|
|
EDT_DeleteTableCells(context, 1);
|
|
fe_EditorUpdateToolbar(context, 0);
|
|
}
|
|
}
|
|
|
|
Boolean
|
|
fe_EditorTableCellGetData(MWContext* context, EDT_TableCellData* p_data)
|
|
{
|
|
EDT_TableCellData* cell_data = EDT_GetTableCellData(context);
|
|
if (cell_data) {
|
|
*p_data = *cell_data;
|
|
if (cell_data->pColorBackground != NULL) {
|
|
static LO_Color color_buf;
|
|
color_buf = *cell_data->pColorBackground;
|
|
p_data->pColorBackground = &color_buf; /*caller cannot hang on*/
|
|
}
|
|
EDT_FreeTableCellData(cell_data);
|
|
return TRUE;
|
|
} else {
|
|
return FALSE;
|
|
}
|
|
}
|
|
|
|
void
|
|
fe_EditorTableCellSetData(MWContext* context, EDT_TableCellData* p_data)
|
|
{
|
|
EDT_TableCellData* cell_data = EDT_GetTableCellData(context);
|
|
if (cell_data) {
|
|
COPY_SAVE_EXTRA(cell_data, p_data);
|
|
DUP_BGCOLOR(cell_data, p_data);
|
|
EDT_SetTableCellData(context, cell_data);
|
|
EDT_FreeTableCellData(cell_data);
|
|
fe_EditorUpdateToolbar(context, 0);
|
|
}
|
|
}
|
|
|
|
void
|
|
fe_EditorPublish(MWContext* context)
|
|
{
|
|
/* Get the destination and what files to include from the user:*/
|
|
fe_EditorPublishDialogDo(context);
|
|
}
|
|
|
|
void
|
|
fe_EditorDisplayTablesSet(MWContext* context, Boolean display)
|
|
{
|
|
EDT_SetDisplayTables(context, display == TRUE);
|
|
fe_EditorUpdateToolbar(context, 0);
|
|
}
|
|
|
|
Boolean
|
|
fe_EditorDisplayTablesGet(MWContext* context)
|
|
{
|
|
return (EDT_GetDisplayTables(context) == TRUE);
|
|
}
|
|
|
|
static char*
|
|
fe_action_name(char* buf,
|
|
XtActionsRec* action_list,
|
|
Cardinal action_list_size,
|
|
XtActionProc action_proc,
|
|
String *av, Cardinal *ac)
|
|
{
|
|
int i;
|
|
char* name = XP_GetString(XFE_UNKNOWN); /* "<unknown>" */
|
|
|
|
if (action_list_size == 0)
|
|
action_list_size = 100000; /* look for null termination */
|
|
|
|
/*
|
|
* Look for action name in the action_list.
|
|
*/
|
|
for (i = 0; i < action_list_size && action_list[i].string != NULL; i++) {
|
|
if (action_list[i].proc == action_proc) {
|
|
name = action_list[i].string;
|
|
break;
|
|
}
|
|
}
|
|
|
|
strcpy(buf, name);
|
|
strcat(buf, "(");
|
|
for (i = 0; i < *ac; i++) {
|
|
if (i != 0)
|
|
strcat(buf, ",");
|
|
strcat(buf, av[i]);
|
|
}
|
|
strcat(buf, ")");
|
|
|
|
return buf;
|
|
}
|
|
|
|
#ifdef DEBUG_akkana_not
|
|
#define DEBUG_TABLE_SELECTION
|
|
#endif /* DEBUG_akkana */
|
|
|
|
static GC getXorGC(MWContext* context)
|
|
{
|
|
XGCValues gcv;
|
|
|
|
#ifdef DEBUG_TABLE_SELECTION
|
|
printf("getting new XOR GC\n");
|
|
#endif /* DEBUG_TABLE_SELECTION */
|
|
|
|
#if FEEDBACK_WITH_TABLE_BORDER_COLOR
|
|
/* colors come back as 0, 0, 0 and don't show up */
|
|
if (edElement->lo_any.type == LO_TABLE)
|
|
{
|
|
LO_TableStruct* ts = &(edElement->lo_table);
|
|
gcv.foreground = fe_GetPixel(context,
|
|
ts->border_color.red,
|
|
ts->border_color.green,
|
|
ts->border_color.blue);
|
|
}
|
|
else
|
|
#endif /* FEEDBACK_WITH_TABLE_BORDER_COLOR */
|
|
gcv.foreground = fe_GetPixel(context, 0xff, 0xff, 0xff);
|
|
gcv.function = GXxor;
|
|
return fe_GetGC(CONTEXT_WIDGET(context), GCForeground|GCFunction, &gcv);
|
|
}
|
|
|
|
void xfe_GetShiftAndCtrl(XEvent* event, Boolean* shiftP, Boolean* ctrlP)
|
|
{
|
|
if (event->type == KeyPress || event->type == KeyRelease) {
|
|
unsigned int mask = event->xkey.state;
|
|
if (shiftP) *shiftP = ((mask & ShiftMask) != 0);
|
|
if (ctrlP) *ctrlP = ((mask & ControlMask) != 0);
|
|
}
|
|
else if (event->type == ButtonPress || event->type == ButtonRelease) {
|
|
unsigned int mask = event->xbutton.state;
|
|
if (shiftP) *shiftP = ((mask & ShiftMask) != 0);
|
|
if (ctrlP) *ctrlP = ((mask & ControlMask) != 0);
|
|
}
|
|
}
|
|
|
|
/* TEMPORARY HACK ALERT!!!
|
|
* The backend doesn't keep track of which cell in a table is selected
|
|
* for purposes of transition from within-cell selection extend to
|
|
* inter-cell selection, and relies on the front-end to handle that.
|
|
* But that means that each editor instance needs to keep track of the
|
|
* selected cell (if any), which means this code would have to live in
|
|
* EditorView.cpp. But that means a lot of code change which I won't
|
|
* be able to check in for quite some time due to the stability freeze.
|
|
* So as a temporary measure, they're implemented as statics and we
|
|
* count on the unliklihood of selection-extending in multiple windows
|
|
* at once. (Safe for drag-select, less so for shift-ctrl-click.) Yuck.
|
|
*/
|
|
static LO_Element* edElement = 0;
|
|
static XP_Bool crossedCellBoundary = FALSE;
|
|
static XP_Bool m_resizingTable = FALSE;
|
|
static XP_Rect m_sizingRect;
|
|
static GC m_gc = 0;
|
|
|
|
|
|
/*
|
|
* NOTE: because of this routine and its explicit need to disown the
|
|
* primary selection, we can't select text in the editor and then
|
|
* middle mouse paste it into another location in the same window...
|
|
*
|
|
* - any ideas on how we might get around this one?...
|
|
*/
|
|
/* fe_EditorGrabFocus is called on starting a selection,
|
|
* and when extending a selection with control-left-click
|
|
* (followed then by fe_EditorSelectionExtend).
|
|
*/
|
|
void
|
|
fe_EditorGrabFocus(MWContext* context, XEvent *event)
|
|
{
|
|
Widget da = NULL;
|
|
XEvent ev;
|
|
unsigned long x, y;
|
|
Time time;
|
|
LO_Element* leHit = 0;
|
|
ED_HitType iTableHit;
|
|
|
|
#ifdef DEBUG_TABLE_SELECTION
|
|
printf("fe_EditorGrabFocus\n");
|
|
#endif /* DEBUG_TABLE_SELECTION */
|
|
|
|
#ifdef DREDD_EDITOR
|
|
fe_UserActivity(context); /* tell the app who has focus */
|
|
#else
|
|
|
|
da = CONTEXT_DATA(context)->drawing_area;
|
|
|
|
ev.type = FocusIn;
|
|
ev.xfocus.type = FocusIn;
|
|
ev.xfocus.send_event = False;
|
|
ev.xfocus.display = XtDisplay(da);
|
|
ev.xfocus.window = XtWindow(da);
|
|
ev.xfocus.mode = NotifyNormal;
|
|
ev.xfocus.detail = NotifyDetailNone;
|
|
|
|
XtDispatchEvent(&ev);
|
|
|
|
XSync(XtDisplay(da), False);
|
|
|
|
/*
|
|
XtSetKeyboardFocus(CONTEXT_WIDGET(context), da);
|
|
*/
|
|
#endif
|
|
|
|
/* don't do this -- Japanese input method requires focus
|
|
fe_NeutralizeFocus(context);
|
|
*/
|
|
|
|
time = fe_getTimeFromEvent(context, event); /* get the time */
|
|
|
|
fe_EventLOCoords(context, event, &x, &y); /* get the doc co-ords */
|
|
|
|
/* see if we're in a table first... */
|
|
iTableHit = EDT_GetTableHitRegion(context, x, y, &leHit, False);
|
|
if (iTableHit == ED_HIT_SIZE_TABLE_WIDTH
|
|
|| iTableHit == ED_HIT_SIZE_TABLE_HEIGHT
|
|
|| iTableHit == ED_HIT_SIZE_COL || iTableHit == ED_HIT_SIZE_ROW
|
|
|| iTableHit == ED_HIT_ADD_ROWS || iTableHit == ED_HIT_ADD_COLS)
|
|
{
|
|
#ifdef DEBUG_TABLE_SELECTION
|
|
printf("resizing a table\n");
|
|
#endif /* DEBUG_TABLE_SELECTION */
|
|
if( EDT_StartSizing(context, leHit, x, y, FALSE, &m_sizingRect) )
|
|
{
|
|
Drawable drawable = CONTEXT_DATA(context)->drawable->xdrawable;
|
|
|
|
m_resizingTable = TRUE;
|
|
edElement = leHit;
|
|
|
|
/* Begin the sizing feedback: */
|
|
m_gc = getXorGC(context);
|
|
XDrawRectangle(XtDisplay(CONTEXT_WIDGET(context)), drawable, m_gc,
|
|
m_sizingRect.left, m_sizingRect.top,
|
|
m_sizingRect.right - m_sizingRect.left,
|
|
m_sizingRect.bottom - m_sizingRect.top);
|
|
#ifdef DEBUG_TABLE_SELECTION
|
|
printf("fe_EditorGrabFocus: sizing rect (%d - %d) x (%d - %d)\n",
|
|
m_sizingRect.left, m_sizingRect.right,
|
|
m_sizingRect.bottom, m_sizingRect.top);
|
|
#endif /* DEBUG_TABLE_SELECTION */
|
|
}
|
|
#ifdef DEBUG_TABLE_SELECTION
|
|
else
|
|
printf("EDT_StartSizing returned 0\n");
|
|
#endif /* DEBUG_TABLE_SELECTION */
|
|
return;
|
|
}
|
|
m_resizingTable = FALSE;
|
|
|
|
/* Do this first, so the EDT_ClearSelection() in the call doesn't
|
|
* blow away what EDT_StartSelection() does!
|
|
*/
|
|
fe_EditorDisownSelection(context, time, False); /* give up selection */
|
|
|
|
if ( iTableHit != ED_HIT_NONE )
|
|
{
|
|
Boolean shift, ctrl;
|
|
xfe_GetShiftAndCtrl(event, &shift, &ctrl);
|
|
if (iTableHit == ED_HIT_SEL_TABLE && ctrl)
|
|
iTableHit = ED_HIT_SEL_ALL_CELLS;
|
|
/* SUPPORT_DRAG_TABLE_ELEMENTS: as long as the XFE doesn't support
|
|
* dragging table elements, it might be nice to make it easier for users
|
|
* by mapping ED_HIT_DRAG_TABLE (which is used as the shorthand for
|
|
* select-element-to-drag) to something more useful like cell selection.
|
|
* But this needs more work.
|
|
*/
|
|
#define SUPPORT_DRAG_TABLE_ELEMENTS
|
|
#ifndef SUPPORT_DRAG_TABLE_ELEMENTS
|
|
else if (iTableHit == ED_HIT_DRAG_TABLE)
|
|
iTableHit = ED_HIT_SEL_CELL;
|
|
#endif /* SUPPORT_DRAG_TABLE_ELEMENTS */
|
|
#ifdef DEBUG_TABLE_SELECTION
|
|
printf("iTableHit: %d\n", iTableHit);
|
|
#endif /* DEBUG_TABLE_SELECTION */
|
|
|
|
if (iTableHit == ED_HIT_SEL_CELL)
|
|
edElement = leHit;
|
|
else
|
|
edElement = 0;
|
|
EDT_SelectTableElement(context, x, y,
|
|
leHit, iTableHit,
|
|
ctrl, /* bModifierKeyPressed */
|
|
FALSE); /* bExtendSelection */
|
|
}
|
|
/* EDT_GetTableHitRegion returns ED_HIT_NONE inside a cell */
|
|
else if (leHit && leHit->type == LO_CELL)
|
|
{
|
|
/* Two cases: we might be starting a new selection (click alone),
|
|
* or we might be extending the selection with control-click.
|
|
*/
|
|
Boolean shift, ctrl;
|
|
xfe_GetShiftAndCtrl(event, &shift, &ctrl);
|
|
#ifdef DEBUG_TABLE_SELECTION
|
|
printf("cell: edElement = 0x%x, shift = %d, ctrl = %d\n",
|
|
edElement, shift, ctrl);
|
|
#endif /* DEBUG_TABLE_SELECTION */
|
|
|
|
/* Reset crossedCellBoundary if we're in a different table
|
|
* from the old edElement, or if there was no old edElement.
|
|
*/
|
|
if (!ctrl || !edElement
|
|
|| (lo_GetParentTable(context, edElement)
|
|
!= lo_GetParentTable(context, leHit)))
|
|
{
|
|
edElement = leHit;
|
|
crossedCellBoundary = FALSE;
|
|
}
|
|
else
|
|
{
|
|
/* Control-selection to add cell to existing selection:
|
|
* select old cell as well as new one.
|
|
*/
|
|
#ifdef DEBUG_TABLE_SELECTION
|
|
printf("control-selection. edElement = 0x%x\n", edElement);
|
|
#endif /* DEBUG_TABLE_SELECTION */
|
|
if (edElement)
|
|
EDT_SelectTableElement(context, x, y,
|
|
edElement, ED_HIT_SEL_CELL,
|
|
ctrl, /* bModifierKeyPressed */
|
|
FALSE); /* bExtendSelection */
|
|
EDT_SelectTableElement(context, x, y,
|
|
leHit, ED_HIT_SEL_CELL,
|
|
ctrl, /* bModifierKeyPressed */
|
|
FALSE); /* bExtendSelection */
|
|
edElement = 0;
|
|
crossedCellBoundary = TRUE;
|
|
return;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
int i;
|
|
/* See if we're on an element which can resize, like an image */
|
|
edElement = LO_XYToElement(context, (int32)x, (int32)y, NULL);
|
|
if ((i = EDT_CanSizeObject(context, edElement, x, y)) != ED_SIZE_NONE)
|
|
{
|
|
#ifdef DEBUG_TABLE_SELECTION
|
|
printf("resizing a non-table element\n");
|
|
#endif /* DEBUG_TABLE_SELECTION */
|
|
/* Begin the sizing feedback: */
|
|
m_gc = getXorGC(context);
|
|
if( EDT_StartSizing(context, edElement, x, y, FALSE, &m_sizingRect) )
|
|
{
|
|
XDrawRectangle(XtDisplay(CONTEXT_WIDGET(context)),
|
|
CONTEXT_DATA(context)->drawable->xdrawable,
|
|
m_gc,
|
|
m_sizingRect.left, m_sizingRect.top,
|
|
m_sizingRect.right - m_sizingRect.left,
|
|
m_sizingRect.bottom - m_sizingRect.top);
|
|
m_resizingTable = TRUE;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
edElement = 0;
|
|
}
|
|
crossedCellBoundary = FALSE;
|
|
}
|
|
|
|
EDT_StartSelection(context, x, y);
|
|
|
|
#ifdef ENDER
|
|
/* See if we're in an HTML textarea -- if so, we need to
|
|
* fiddle with toolbars to indicate which area has focus.
|
|
* This has to happen *after* EDT_StartSelection
|
|
* so the EditorView knows where the cursor has moved.
|
|
*/
|
|
if (EDITOR_CONTEXT_DATA(context)->embedded)
|
|
XFE_EmbeddedEditorViewFocus(context);
|
|
#endif /* ENDER */
|
|
}
|
|
|
|
/* We never call fe_EditorSelectionBegin(), as far as I can tell. ...Akk */
|
|
void
|
|
fe_EditorSelectionBegin(MWContext* context, XEvent *event)
|
|
{
|
|
unsigned long x, y;
|
|
Time time;
|
|
LO_Element* leHit = 0;
|
|
ED_HitType iTableHit;
|
|
|
|
#ifdef DEBUG_TABLE_SELECTION
|
|
printf("fe_EditorSelectionBegin\n");
|
|
#endif /* DEBUG_TABLE_SELECTION */
|
|
|
|
edElement = 0;
|
|
crossedCellBoundary = FALSE;
|
|
|
|
/* don't do this -- Japanese input method requires focus
|
|
fe_NeutralizeFocus (context);
|
|
*/
|
|
|
|
if (CONTEXT_DATA(context)->editor.ascroll_data.timer_id) {
|
|
XtRemoveTimeOut(CONTEXT_DATA(context)->editor.ascroll_data.timer_id);
|
|
CONTEXT_DATA(context)->editor.ascroll_data.timer_id = 0;
|
|
}
|
|
|
|
if (CONTEXT_DATA(context)->clicking_blocked
|
|
|| CONTEXT_DATA (context)->synchronous_url_dialog)
|
|
{
|
|
fe_Bell(context);
|
|
return;
|
|
}
|
|
|
|
time = fe_getTimeFromEvent(context, event); /* get the time */
|
|
|
|
fe_EventLOCoords(context, event, &x, &y);
|
|
|
|
#ifdef DONT_rhess
|
|
/*
|
|
* WARNING... [ nuke the primary selection owner ]
|
|
*/
|
|
{
|
|
Display *dpy = XtDisplay(CONTEXT_WIDGET(context));
|
|
Window owner = XGetSelectionOwner(dpy, XA_PRIMARY);
|
|
|
|
if (owner != None) {
|
|
XSetSelectionOwner(dpy, XA_PRIMARY, None, time);
|
|
}
|
|
}
|
|
#endif
|
|
|
|
/*
|
|
* Do this first, so the EDT_ClearSelection() in the call doesn't
|
|
* blow away what EDT_StartSelection() does!
|
|
*/
|
|
fe_EditorDisownSelection(context, time, False);
|
|
|
|
/* see if we're in a table first... */
|
|
iTableHit = EDT_GetTableHitRegion(context, x, y, &leHit, False);
|
|
if ( iTableHit != ED_HIT_NONE )
|
|
{
|
|
Boolean shift, ctrl;
|
|
xfe_GetShiftAndCtrl(event, &shift, &ctrl);
|
|
EDT_SelectTableElement(context, x, y,
|
|
leHit, iTableHit,
|
|
ctrl, /* bModifierKeyPressed */
|
|
shift); /*bExtendSelection*/
|
|
edElement = leHit;
|
|
return;
|
|
}
|
|
|
|
/* EDT_GetTableHitRegion returns ED_HIT_NONE in a cell */
|
|
if (leHit && leHit->type == LO_CELL)
|
|
edElement = leHit;
|
|
|
|
/* need to implement disown */
|
|
EDT_StartSelection(context, x, y);
|
|
}
|
|
|
|
/* This is called on dragging, and also on shift-click */
|
|
void
|
|
fe_EditorSelectionExtend(MWContext* context, XEvent *event)
|
|
{
|
|
Time time;
|
|
unsigned long x;
|
|
unsigned long y;
|
|
LO_Element* leHit = 0;
|
|
ED_HitType iTableHit;
|
|
|
|
#ifdef DEBUG_TABLE_SELECTION
|
|
printf("fe_EditorSelectionExtend\n");
|
|
#endif /* DEBUG_TABLE_SELECTION */
|
|
|
|
time = fe_getTimeFromEvent(context, event); /* get the time */
|
|
|
|
/* don't do this -- Japanese input method requires focus
|
|
fe_NeutralizeFocus(context);
|
|
*/
|
|
|
|
fe_EventLOCoords (context, event, &x, &y);
|
|
|
|
if (CONTEXT_DATA(context)->editor.ascroll_data.timer_id) {
|
|
XtRemoveTimeOut(CONTEXT_DATA(context)->editor.ascroll_data.timer_id);
|
|
CONTEXT_DATA(context)->editor.ascroll_data.timer_id = 0;
|
|
}
|
|
|
|
fe_editor_keep_pointer_visible(context, x, y);
|
|
|
|
if (m_resizingTable)
|
|
{
|
|
Drawable drawable = CONTEXT_DATA(context)->drawable->xdrawable;
|
|
XDrawRectangle(XtDisplay(CONTEXT_WIDGET(context)), drawable, m_gc,
|
|
m_sizingRect.left, m_sizingRect.top,
|
|
m_sizingRect.right - m_sizingRect.left,
|
|
m_sizingRect.bottom - m_sizingRect.top);
|
|
#ifdef DEBUG_TABLE_SELECTION
|
|
printf("fe_EditorSelectionExtend: sizing rect (%d - %d) x (%d - %d)\n",
|
|
m_sizingRect.left, m_sizingRect.right,
|
|
m_sizingRect.bottom, m_sizingRect.top);
|
|
#endif /* DEBUG_TABLE_SELECTION */
|
|
|
|
/* Don't check the return value from EDT_GetSizingRect;
|
|
* sometimes CSizingObject::GetSizingRect says the rect
|
|
* hasn't changed, and returns FALSE, which unfortunately is
|
|
* the same as CEditBuf::GetSizingRect's error code that
|
|
* we're not in the middle of a sizing operation.
|
|
*/
|
|
EDT_GetSizingRect(context, x, y, FALSE, &m_sizingRect);
|
|
|
|
#ifdef DEBUG_TABLE_SELECTION
|
|
printf(" : sizing rect (%d - %d) x (%d - %d)\n",
|
|
m_sizingRect.left, m_sizingRect.right,
|
|
m_sizingRect.bottom, m_sizingRect.top);
|
|
#endif /* DEBUG_TABLE_SELECTION */
|
|
/* show feedback */
|
|
XDrawRectangle(XtDisplay(CONTEXT_WIDGET(context)), drawable, m_gc,
|
|
m_sizingRect.left, m_sizingRect.top,
|
|
m_sizingRect.right - m_sizingRect.left,
|
|
m_sizingRect.bottom - m_sizingRect.top);
|
|
return;
|
|
}
|
|
|
|
#ifdef DONT_rhess
|
|
/*
|
|
* WARNING... [ nuke the primary selection owner ]
|
|
*/
|
|
{
|
|
Display *dpy = XtDisplay(CONTEXT_WIDGET(context));
|
|
Window owner = XGetSelectionOwner(dpy, XA_PRIMARY);
|
|
|
|
if (owner != None) {
|
|
XSetSelectionOwner(dpy, XA_PRIMARY, None, time);
|
|
}
|
|
}
|
|
#endif
|
|
|
|
/* see if we're in a table first... */
|
|
iTableHit = EDT_GetTableHitRegion(context, x, y, &leHit, False);
|
|
|
|
if (leHit && leHit->type == LO_CELL)
|
|
{
|
|
if (edElement && leHit != edElement)
|
|
{
|
|
Boolean shift, ctrl;
|
|
xfe_GetShiftAndCtrl(event, &shift, &ctrl);
|
|
|
|
if (!crossedCellBoundary)
|
|
{
|
|
fe_EditorDisownSelection(context, time, False);
|
|
EDT_SelectTableElement(context, x, y,
|
|
edElement, ED_HIT_SEL_CELL,
|
|
FALSE, /* bModifierKeyPressed */
|
|
FALSE); /*bExtendSelection*/
|
|
crossedCellBoundary = TRUE;
|
|
}
|
|
|
|
EDT_SelectTableElement(context, x, y,
|
|
leHit, ED_HIT_SEL_CELL,
|
|
ctrl, /* bModifierKeyPressed */
|
|
TRUE); /*bExtendSelection*/
|
|
}
|
|
else
|
|
{
|
|
/* edElement = leHit; */
|
|
EDT_ExtendSelection(context, x, y);
|
|
}
|
|
}
|
|
|
|
else
|
|
/* need to implement own */
|
|
EDT_ExtendSelection(context, x, y);
|
|
|
|
#ifdef DONT_rhess
|
|
fe_EditorOwnSelection (context, time, False, False);
|
|
#endif
|
|
|
|
#if 0
|
|
/* Making a selection turns off "Save Next" mode. */
|
|
if (CONTEXT_DATA (context)->save_next_mode_p)
|
|
{
|
|
XBell (XtDisplay (widget), 0);
|
|
CONTEXT_DATA (context)->save_next_mode_p = False;
|
|
fe_SetCursor (context, False);
|
|
XFE_Progress (context, fe_globalData.click_to_save_cancelled_message);
|
|
}
|
|
#endif
|
|
}
|
|
|
|
void
|
|
fe_EditorSelectionEnd(MWContext *context, XEvent *event)
|
|
{
|
|
Time time;
|
|
unsigned long x;
|
|
unsigned long y;
|
|
LO_Element* leHit = 0;
|
|
ED_HitType iTableHit;
|
|
Boolean shift, ctrl;
|
|
|
|
#ifdef DEBUG_TABLE_SELECTION
|
|
printf("fe_EditorSelectionEnd\n");
|
|
#endif /* DEBUG_TABLE_SELECTION */
|
|
|
|
fe_UserActivity(context);
|
|
time = fe_getTimeFromEvent(context, event); /* get the time */
|
|
|
|
fe_EventLOCoords (context, event, &x, &y);
|
|
|
|
if (CONTEXT_DATA(context)->editor.ascroll_data.timer_id) {
|
|
XtRemoveTimeOut(CONTEXT_DATA(context)->editor.ascroll_data.timer_id);
|
|
CONTEXT_DATA(context)->editor.ascroll_data.timer_id = 0;
|
|
}
|
|
|
|
if (m_resizingTable)
|
|
{
|
|
Drawable drawable = CONTEXT_DATA(context)->drawable->xdrawable;
|
|
/* clear resizing feedback */
|
|
XDrawRectangle(XtDisplay(CONTEXT_WIDGET(context)), drawable, m_gc,
|
|
m_sizingRect.left, m_sizingRect.top,
|
|
m_sizingRect.right - m_sizingRect.left,
|
|
m_sizingRect.bottom - m_sizingRect.top);
|
|
/*
|
|
* XXX After resizing an object then leaving the
|
|
* mouse at the same position where the button was released,
|
|
* calling LO_XYToElement will return a different object
|
|
* (maybe a border?) which is not resizable.
|
|
* So try to turn off the cursor after resizing so the
|
|
* user doesn't think it's still draggable.
|
|
*/
|
|
fe_SetCursor (context, False);
|
|
|
|
#ifdef DEBUG_TABLE_SELECTION
|
|
printf("Ending table resize: [%d, %d, %d, %d]\n",
|
|
m_sizingRect.left, m_sizingRect.right, m_sizingRect.top,
|
|
m_sizingRect.bottom);
|
|
#endif /* DEBUG_TABLE_SELECTION */
|
|
EDT_EndSizing(context);
|
|
m_resizingTable = FALSE;
|
|
return;
|
|
}
|
|
|
|
/* see if we're in a table first... */
|
|
iTableHit = EDT_GetTableHitRegion( context, x, y, &leHit, FALSE );
|
|
xfe_GetShiftAndCtrl(event, &shift, &ctrl);
|
|
#ifdef DEBUG_TABLE_SELECTION
|
|
printf("iTableHit = %d\n", iTableHit);
|
|
#endif /* DEBUG_TABLE_SELECTION */
|
|
if (iTableHit == ED_HIT_SEL_TABLE && ctrl)
|
|
iTableHit = ED_HIT_SEL_ALL_CELLS;
|
|
#ifndef SUPPORT_DRAG_TABLE_ELEMENTS
|
|
else if (iTableHit == ED_HIT_DRAG_TABLE)
|
|
iTableHit = ED_HIT_SEL_CELL;
|
|
#endif /* SUPPORT_DRAG_TABLE_ELEMENTS */
|
|
#ifdef DEBUG_TABLE_SELECTION
|
|
printf(" ... iTableHit = %d\n", iTableHit);
|
|
#endif /* DEBUG_TABLE_SELECTION */
|
|
if ( iTableHit == ED_HIT_SEL_TABLE
|
|
|| iTableHit == ED_HIT_SEL_CELL
|
|
|| iTableHit == ED_HIT_SEL_ALL_CELLS
|
|
|| iTableHit == ED_HIT_SEL_COL
|
|
|| iTableHit == ED_HIT_SEL_ROW
|
|
|| iTableHit == ED_HIT_DRAG_TABLE )
|
|
{
|
|
if (!ctrl) /* don't double-toggle the selection */
|
|
EDT_SelectTableElement(context, x, y,
|
|
leHit, iTableHit,
|
|
ctrl, /* bModifierKeyPressed */
|
|
shift); /*bExtendSelection*/
|
|
}
|
|
else
|
|
EDT_EndSelection(context, x, y);
|
|
|
|
fe_EditorOwnSelection(context, time, False, False);
|
|
}
|
|
|
|
#define XFE_EDITOR_MAX_INSERT_SIZE 512
|
|
|
|
void
|
|
fe_EditorKeyInsert(MWContext* context, Widget widget, XEvent* event)
|
|
{
|
|
char buf[XFE_EDITOR_MAX_INSERT_SIZE + 1];
|
|
KeySym keysym;
|
|
Status status_return;
|
|
int len;
|
|
EDT_ClipboardResult result = EDT_COP_OK;
|
|
INTL_CharSetInfo c = LO_GetDocumentCharacterSetInfo(context);
|
|
|
|
if (widget != CONTEXT_DATA(context)->drawing_area) {
|
|
#ifdef DEBUG
|
|
(void) fprintf(real_stderr,
|
|
"fe_editor_key_input_action got wrong widget (%s)\n",
|
|
XtName(widget));
|
|
#endif /* DEBUG */
|
|
XtSetKeyboardFocus(CONTEXT_WIDGET(context),
|
|
CONTEXT_DATA(context)->drawing_area);
|
|
return;
|
|
}
|
|
len = XmImMbLookupString(widget,
|
|
(XKeyEvent *)event,
|
|
buf,
|
|
XFE_EDITOR_MAX_INSERT_SIZE,
|
|
&keysym,
|
|
&status_return);
|
|
|
|
if (status_return == XBufferOverflow || len > XFE_EDITOR_MAX_INSERT_SIZE)
|
|
return;
|
|
|
|
if (len > 0) {
|
|
char *str;
|
|
|
|
buf[len] = 0;
|
|
str = (char *) fe_ConvertFromLocaleEncoding(INTL_GetCSIWinCSID(c),
|
|
(unsigned char *) buf);
|
|
if (str != buf) {
|
|
len = strlen(str);
|
|
}
|
|
|
|
if (len == 1) {
|
|
result = EDT_KeyDown(context, str[0], 0, 0);
|
|
} else if (len > 1) {
|
|
result = EDT_PasteText(context, str);
|
|
}
|
|
|
|
if (result != EDT_COP_OK) {
|
|
fe_editor_report_copy_error(context, result);
|
|
}
|
|
|
|
if (str != buf && str != NULL) {
|
|
XP_FREE(str);
|
|
}
|
|
}
|
|
}
|
|
|
|
void
|
|
fe_EditorTab(MWContext* context, Boolean forward, Boolean force)
|
|
{
|
|
EDT_ClipboardResult result = EDT_TabKey(context, forward, force);
|
|
|
|
if (result != EDT_COP_OK)
|
|
fe_editor_report_copy_error(context, result);
|
|
}
|
|
|
|
/*
|
|
* Editor specific action handlers.
|
|
*/
|
|
static void
|
|
fe_ActionWrongContextAlert(MWContext*, XtActionProc, String*, Cardinal*);
|
|
|
|
#ifdef DEBUG
|
|
#define FE_ACTION_ALERT(c, p, a, n) fe_ActionWrongContextAlert((c),(p),(a),(n))
|
|
#else
|
|
#define FE_ACTION_ALERT(c, p, a, n)
|
|
#endif
|
|
|
|
#define FE_ACTION_VALIDATE(proc) \
|
|
if (!(context) || !EDT_IS_EDITOR(context)) { \
|
|
FE_ACTION_ALERT(context, (proc), av, ac); \
|
|
return; \
|
|
}
|
|
|
|
|
|
#define FE_SYNTAX_ERROR(proc) fe_ActionSyntaxAlert(context, (proc), av, ac)
|
|
|
|
/*=========================== ACTIONS =================================*/
|
|
static void
|
|
fe_get_selected_text_rect(MWContext* context,
|
|
LO_TextStruct* text,
|
|
int32* rect_x,
|
|
int32* rect_y,
|
|
int32* rect_width,
|
|
int32* rect_height)
|
|
{
|
|
PA_Block text_save = text->text;
|
|
int len_save = text->text_len;
|
|
LO_TextInfo info;
|
|
|
|
text->text_len = text->sel_start;
|
|
XFE_GetTextInfo(context, text, &info);
|
|
|
|
*rect_x = text->x + text->x_offset + info.max_width;
|
|
*rect_y = text->y + text->y_offset;
|
|
|
|
text->text = (PA_Block)((char*)text->text + text->sel_start);
|
|
text->text_len = text->sel_end - text->sel_start;
|
|
|
|
XFE_GetTextInfo(context, text, &info);
|
|
|
|
*rect_width = info.max_width;
|
|
*rect_height = info.ascent + info.descent;
|
|
|
|
text->text = text_save;
|
|
text->text_len = len_save;
|
|
}
|
|
|
|
static Boolean
|
|
fe_editor_selection_contains_point(MWContext* context, int32 x, int32 y)
|
|
{
|
|
int32 start_selection;
|
|
int32 end_selection;
|
|
LO_Element* start_element = NULL;
|
|
LO_Element* end_element = NULL;
|
|
LO_Element* lo_element;
|
|
int32 rect_x;
|
|
int32 rect_y;
|
|
int32 rect_width;
|
|
int32 rect_height;
|
|
CL_Layer *layer;
|
|
|
|
if (!EDT_IsSelected(context))
|
|
return FALSE;
|
|
|
|
LO_GetSelectionEndpoints(context,
|
|
&start_element,
|
|
&end_element,
|
|
&start_selection,
|
|
&end_selection,
|
|
&layer);
|
|
|
|
if (start_element == NULL)
|
|
return FALSE;
|
|
|
|
for (lo_element = start_element;
|
|
lo_element != NULL;
|
|
lo_element = ((LO_Any *)lo_element)->next) {
|
|
|
|
if (lo_element->type == LO_TEXT &&
|
|
(lo_element == start_element || lo_element == end_element)) {
|
|
LO_TextStruct* text = (LO_TextStruct*)lo_element;
|
|
|
|
if (text->text == NULL) {
|
|
if (text->prev != NULL && text->prev->type == LO_TEXT) {
|
|
text = (LO_TextStruct*)text->prev;
|
|
} else {
|
|
text = (LO_TextStruct*)text->next;
|
|
}
|
|
}
|
|
|
|
if (text->text == NULL)
|
|
continue;
|
|
|
|
fe_get_selected_text_rect(context, text,
|
|
&rect_x,
|
|
&rect_y,
|
|
&rect_width,
|
|
&rect_height);
|
|
|
|
} else if (lo_element->type == LO_LINEFEED) {
|
|
continue;
|
|
|
|
} else {
|
|
LO_Any* lo_any = (LO_Any*)lo_element;
|
|
rect_x = lo_any->x + lo_any->x_offset;
|
|
rect_y = lo_any->y + lo_any->y_offset;
|
|
rect_width = lo_any->width;
|
|
rect_height = lo_any->height;
|
|
}
|
|
|
|
if (x > rect_x && y > rect_y &&
|
|
x < (rect_x + rect_width) && y < (rect_y + rect_height))
|
|
return TRUE;
|
|
|
|
if (lo_element == end_element)
|
|
break;
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
Boolean
|
|
fe_EditorSelectionContainsPoint(MWContext* context, int32 x, int32 y)
|
|
{
|
|
return fe_editor_selection_contains_point(context, x, y);
|
|
}
|
|
|
|
static char*
|
|
fe_lo_image_anchor(LO_ImageStruct* image,
|
|
long x, long y,
|
|
MWContext* context)
|
|
{
|
|
if (image->is_icon &&
|
|
image->icon_number == IL_IMAGE_DELAYED) { /* delayed image */
|
|
long width, height;
|
|
|
|
fe_IconSize(IL_IMAGE_DELAYED, &width, &height);
|
|
|
|
if (image->alt &&
|
|
image->alt_len &&
|
|
(x > image->x + image->x_offset + 1 + 4 + width)) {
|
|
if (image->anchor_href) {
|
|
return (char *)image->anchor_href->anchor;
|
|
}
|
|
} /* else fall to end */
|
|
}
|
|
|
|
/*
|
|
* Internal editor image maybe?
|
|
*/
|
|
else if (image->is_icon &&
|
|
(image->icon_number == IL_EDIT_NAMED_ANCHOR ||
|
|
image->icon_number == IL_EDIT_FORM_ELEMENT ||
|
|
image->icon_number == IL_EDIT_UNSUPPORTED_TAG ||
|
|
image->icon_number == IL_EDIT_UNSUPPORTED_END_TAG)) {
|
|
|
|
if (image->alt != NULL)
|
|
return (char*)image->alt;
|
|
else
|
|
return "";
|
|
}
|
|
|
|
/*
|
|
* This would be a client-side usemap image.
|
|
*/
|
|
else if (image->image_attr->usemap_name != NULL) {
|
|
|
|
LO_AnchorData *anchor_href;
|
|
long ix = image->x + image->x_offset;
|
|
long iy = image->y + image->y_offset;
|
|
long mx = x - ix - image->border_width;
|
|
long my = y - iy - image->border_width;
|
|
|
|
anchor_href = LO_MapXYToAreaAnchor(context,
|
|
(LO_ImageStruct *)image,
|
|
mx, my);
|
|
if (anchor_href) {
|
|
return (char *)anchor_href->anchor;
|
|
} /* else fall to end */
|
|
} else { /* some old random image */
|
|
if (image->anchor_href) {
|
|
return (char *)image->anchor_href->anchor;
|
|
}
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
static LO_Element* fe_editor_motion_action_last_le;
|
|
static Boolean fe_editor_motion_action_last_le_is_image;
|
|
static LO_Element* fe_editor_motion_action_last_hit_le;
|
|
static ED_HitType fe_editor_motion_action_last_hit_type;
|
|
|
|
static void
|
|
fe_editor_motion_action(Widget widget, XEvent *event,
|
|
String *av, Cardinal *ac)
|
|
{
|
|
MWContext* context = fe_MotionWidgetToMWContext(widget);
|
|
LO_Element* le;
|
|
LO_Element* leHit;
|
|
ED_HitType iTableHit;
|
|
unsigned long x, y;
|
|
Cursor cursor = None;
|
|
char* progress_string = NULL;
|
|
int status_msg = 0;
|
|
char buf[80];
|
|
char num[32];
|
|
|
|
FE_ACTION_VALIDATE(fe_editor_motion_action);
|
|
|
|
fe_EventLOCoords(context, event, &x, &y); /* get the doc co-ords */
|
|
|
|
/* NOTE: let's see if we're in table stuff first... */
|
|
iTableHit = EDT_GetTableHitRegion( context,
|
|
x, y,
|
|
&leHit,
|
|
FALSE );
|
|
if ( iTableHit ) {
|
|
if ((fe_editor_motion_action_last_hit_type == iTableHit) &&
|
|
(fe_editor_motion_action_last_hit_le == leHit)) {
|
|
return; /* NOTE: no change... */
|
|
}
|
|
|
|
fe_editor_motion_action_last_hit_le = leHit;
|
|
fe_editor_motion_action_last_hit_type = iTableHit;
|
|
|
|
switch ( iTableHit ) {
|
|
case ED_HIT_SEL_TABLE:
|
|
#ifdef DEBUG_TABLE_SELECTION
|
|
#define DEBUG_motion
|
|
#endif
|
|
#ifdef DEBUG_motion
|
|
fprintf(stderr, "ED_HIT_SEL_TABLE:\n");
|
|
#endif
|
|
cursor = CONTEXT_DATA(context)->tab_sel_cursor;
|
|
status_msg = XP_EDT_SEL_TABLE;
|
|
break;
|
|
|
|
case ED_HIT_SEL_ALL_CELLS:
|
|
#ifdef DEBUG_motion
|
|
fprintf(stderr, "ED_HIT_SEL_ALL_CELLS:\n");
|
|
#endif
|
|
cursor = None;
|
|
status_msg = XP_EDT_SEL_ALL_CELLS;
|
|
break;
|
|
|
|
case ED_HIT_SEL_COL:
|
|
#ifdef DEBUG_motion
|
|
fprintf(stderr, "ED_HIT_SEL_COL:\n");
|
|
#endif
|
|
cursor = CONTEXT_DATA(context)->col_sel_cursor;
|
|
status_msg = XP_EDT_SEL_COL;
|
|
break;
|
|
|
|
case ED_HIT_SEL_ROW:
|
|
#ifdef DEBUG_motion
|
|
fprintf(stderr, "ED_HIT_SEL_ROW:\n");
|
|
#endif
|
|
cursor = CONTEXT_DATA(context)->row_sel_cursor;
|
|
status_msg = XP_EDT_SEL_ROW;
|
|
break;
|
|
|
|
case ED_HIT_SEL_CELL:
|
|
#ifdef DEBUG_motion
|
|
fprintf(stderr, "ED_HIT_SEL_CELL:\n");
|
|
#endif
|
|
cursor = CONTEXT_DATA(context)->cel_sel_cursor;
|
|
status_msg = XP_EDT_SEL_CELL;
|
|
break;
|
|
|
|
case ED_HIT_DRAG_TABLE:
|
|
#ifdef DEBUG_motion
|
|
fprintf(stderr, "ED_HIT_DRAG_TABLE:\n");
|
|
#endif
|
|
#ifdef SUPPORT_DRAG_TABLE_ELEMENTS
|
|
cursor = CONTEXT_DATA(context)->resize_tab_cursor;
|
|
status_msg = XP_EDT_DRAG_TABLE;
|
|
#else /* SUPPORT_DRAG_TABLE_ELEMENTS */
|
|
cursor = CONTEXT_DATA(context)->cel_sel_cursor;
|
|
status_msg = XP_EDT_SEL_CELL;
|
|
#endif /* SUPPORT_DRAG_TABLE_ELEMENTS */
|
|
break;
|
|
|
|
case ED_HIT_SIZE_TABLE_WIDTH:
|
|
#ifdef DEBUG_motion
|
|
fprintf(stderr, "ED_HIT_SIZE_TABLE_WIDTH:\n");
|
|
#endif
|
|
cursor = CONTEXT_DATA(context)->resize_tab_cursor;
|
|
status_msg = XP_EDT_SIZE_TABLE_WIDTH;
|
|
break;
|
|
|
|
case ED_HIT_SIZE_TABLE_HEIGHT:
|
|
#ifdef DEBUG_motion
|
|
fprintf(stderr, "ED_HIT_SIZE_TABLE_HEIGHT:\n");
|
|
#endif
|
|
cursor = CONTEXT_DATA(context)->resize_tab_cursor;
|
|
status_msg = XP_EDT_SIZE_TABLE_HEIGHT;
|
|
break;
|
|
|
|
case ED_HIT_SIZE_ROW:
|
|
#ifdef DEBUG_motion
|
|
fprintf(stderr, "ED_HIT_SIZE_ROW:\n");
|
|
#endif
|
|
cursor = CONTEXT_DATA(context)->resize_row_cursor;
|
|
status_msg = XP_EDT_SIZE_ROW;
|
|
break;
|
|
|
|
case ED_HIT_SIZE_COL:
|
|
#ifdef DEBUG_motion
|
|
fprintf(stderr, "ED_HIT_SIZE_COL:\n");
|
|
#endif
|
|
cursor = CONTEXT_DATA(context)->resize_col_cursor;
|
|
status_msg = XP_EDT_SIZE_COL;
|
|
break;
|
|
|
|
case ED_HIT_ADD_ROWS:
|
|
#ifdef DEBUG_motion
|
|
fprintf(stderr, "ED_HIT_ADD_ROWS:\n");
|
|
#endif
|
|
cursor = CONTEXT_DATA(context)->add_row_cursor;
|
|
status_msg = XP_EDT_ADD_ROWS;
|
|
break;
|
|
|
|
case ED_HIT_ADD_COLS:
|
|
#ifdef DEBUG_motion
|
|
fprintf(stderr, "ED_HIT_ADD_COLS:\n");
|
|
#endif
|
|
cursor = CONTEXT_DATA(context)->add_col_cursor;
|
|
status_msg = XP_EDT_ADD_COLS;
|
|
break;
|
|
|
|
default:
|
|
/* NOTE: we shouldn't ever get here... */
|
|
#ifdef DEBUG_motion
|
|
fprintf(stderr, "fe_editor_motion_action default: WARNING ***************\n");
|
|
#endif
|
|
cursor = None;
|
|
}
|
|
if (status_msg != 0)
|
|
progress_string = XP_GetString(status_msg);
|
|
}
|
|
else {
|
|
int sizeType;
|
|
|
|
/* NOTE: let's check for the other elements... */
|
|
|
|
le = LO_XYToElement(context, (int32)x, (int32)y, NULL);
|
|
|
|
/* See if we should show a resize cursor */
|
|
sizeType = EDT_CanSizeObject(context, le, x, y);
|
|
if (sizeType != ED_SIZE_NONE)
|
|
{
|
|
switch(sizeType)
|
|
{
|
|
case ED_SIZE_TOP:
|
|
cursor = CONTEXT_DATA(context)->top_cursor;
|
|
break;
|
|
case ED_SIZE_BOTTOM:
|
|
cursor = CONTEXT_DATA(context)->bottom_cursor;
|
|
break;
|
|
case ED_SIZE_RIGHT:
|
|
cursor = CONTEXT_DATA(context)->right_cursor;
|
|
break;
|
|
case ED_SIZE_LEFT:
|
|
cursor = CONTEXT_DATA(context)->left_cursor;
|
|
break;
|
|
case ED_SIZE_TOP_RIGHT:
|
|
cursor = CONTEXT_DATA(context)->top_right_cursor;
|
|
break;
|
|
case ED_SIZE_BOTTOM_RIGHT:
|
|
cursor = CONTEXT_DATA(context)->bottom_right_cursor;
|
|
break;
|
|
case ED_SIZE_TOP_LEFT:
|
|
cursor = CONTEXT_DATA(context)->top_left_cursor;
|
|
break;
|
|
case ED_SIZE_BOTTOM_LEFT:
|
|
cursor = CONTEXT_DATA(context)->bottom_left_cursor;
|
|
break;
|
|
}
|
|
}
|
|
|
|
else
|
|
{
|
|
if (le == fe_editor_motion_action_last_le &&
|
|
!fe_editor_motion_action_last_le_is_image &&
|
|
(fe_editor_motion_action_last_hit_type == 0)) {
|
|
return; /* NOTE: no change... */
|
|
}
|
|
|
|
fe_editor_motion_action_last_hit_le = NULL;
|
|
fe_editor_motion_action_last_hit_type = 0;
|
|
|
|
fe_editor_motion_action_last_le_is_image = FALSE;
|
|
fe_editor_motion_action_last_le = le;
|
|
|
|
if (le != NULL) {
|
|
/* Are we over text? Special cursor */
|
|
if (le->type == LO_TEXT) {
|
|
cursor = CONTEXT_DATA(context)->editable_text_cursor;
|
|
if (le->lo_text.anchor_href) { /* link */
|
|
progress_string = (char *)le->lo_text.anchor_href->anchor;
|
|
}
|
|
/* Over image, special progress */
|
|
}
|
|
else {
|
|
if (le->type == LO_IMAGE) {
|
|
progress_string = fe_lo_image_anchor(&le->lo_image,
|
|
/* link ? */
|
|
x, y,
|
|
context);
|
|
if (progress_string != NULL && progress_string[0] == '\0')
|
|
progress_string = NULL;
|
|
|
|
if (progress_string == NULL) {
|
|
|
|
if (le->lo_image.image_url) {
|
|
progress_string = (char*)le->lo_image.image_url;
|
|
} else if (le->lo_image.lowres_image_url) {
|
|
progress_string = (char*)le->lo_image.lowres_image_url;
|
|
}
|
|
|
|
if (progress_string != NULL) {
|
|
sprintf(num, ",%d,%d",
|
|
x - le->lo_image.x, y - le->lo_image.y);
|
|
FE_CondenseURL(buf, progress_string,
|
|
sizeof(buf) - 1 - strlen(num));
|
|
strcat(buf, num);
|
|
progress_string = buf;
|
|
fe_editor_motion_action_last_le_is_image = TRUE;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if (CONTEXT_DATA(context)->clicking_blocked ||
|
|
CONTEXT_DATA(context)->synchronous_url_dialog) {
|
|
cursor = CONTEXT_DATA(context)->busy_cursor;
|
|
}
|
|
|
|
if (CONTEXT_DATA(context)->drawing_area) {
|
|
XDefineCursor(XtDisplay(CONTEXT_DATA(context)->drawing_area),
|
|
XtWindow(CONTEXT_DATA(context)->drawing_area),
|
|
cursor);
|
|
}
|
|
|
|
if (progress_string == NULL)
|
|
progress_string = "";
|
|
|
|
if (CONTEXT_DATA(context)->active_url_count == 0) {
|
|
/* If there are transfers in progress, don't document the URL under
|
|
the mouse, since that message would interfere with the transfer
|
|
messages. Do change the cursor, however. */
|
|
fe_MidTruncatedProgress(context, progress_string);
|
|
}
|
|
}
|
|
|
|
#define EA_PREFIX "editor-"
|
|
|
|
static XtActionsRec fe_editor_actions [] =
|
|
{
|
|
{ EA_PREFIX "motion", fe_editor_motion_action },
|
|
};
|
|
|
|
static void
|
|
fe_ActionWrongContextAlert(MWContext* context,
|
|
XtActionProc action_proc,
|
|
String* av, Cardinal* ac)
|
|
{
|
|
char buf[1024];
|
|
char name[1024];
|
|
|
|
fe_action_name(name,
|
|
fe_editor_actions, countof(fe_editor_actions),
|
|
action_proc, av, ac);
|
|
sprintf(buf, XP_GetString(XFE_ACTION_WRONG_CONTEXT), name);
|
|
|
|
if (context)
|
|
FE_Alert(context, buf);
|
|
else
|
|
fprintf(real_stderr, buf);
|
|
}
|
|
|
|
void
|
|
fe_EditorInitActions()
|
|
{
|
|
XtAppAddActions(fe_XtAppContext,
|
|
fe_editor_actions,
|
|
countof(fe_editor_actions)
|
|
);
|
|
}
|
|
|
|
static void
|
|
fe_EditURLCallback(const char* urlToOpen)
|
|
{
|
|
/*
|
|
* Try to get editor first. XP_FindSomeContext() does not look for
|
|
* editor context.
|
|
*/
|
|
MWContext *context = XP_FindContextOfType(NULL, MWContextEditor);
|
|
if (!context) {
|
|
context = XP_FindSomeContext();
|
|
}
|
|
|
|
/* This shouldn't happen. */
|
|
if (!context) {
|
|
XP_ASSERT(0);
|
|
return;
|
|
}
|
|
|
|
/*
|
|
* If urlToOpen is already being edited, this will just pop it to
|
|
* front.
|
|
*/
|
|
fe_EditorEdit(context, NULL, NULL, (char*)urlToOpen);
|
|
}
|
|
|
|
void
|
|
fe_EditorStaticInit()
|
|
{
|
|
EDTPLUG_RegisterEditURLCallback(&fe_EditURLCallback);
|
|
}
|
|
|
|
/*
|
|
* Note: this routine is untested. It's called on dragging to
|
|
* create new rows or columns,
|
|
* not for highlighting selected rows/columns.
|
|
*/
|
|
void
|
|
FE_DisplayAddRowOrColBorder(MWContext* context, XP_Rect *pRect, XP_Bool bErase)
|
|
{
|
|
GC gc;
|
|
XGCValues gcv;
|
|
unsigned long gc_flags;
|
|
fe_Drawable *fe_drawable = CONTEXT_DATA(context)->drawable;
|
|
|
|
#ifdef DEBUG_TABLE_SELECTION
|
|
printf("FE_DisplayAddRowOrColBorder(bErase=%d)\n", bErase);
|
|
#endif /* DEBUG_TABLE_SELECTION */
|
|
|
|
gc_flags = GCForeground | GCLineStyle | GCFunction /*| GCLineWidth */;
|
|
gcv.foreground = 0xFFFFFF;
|
|
gcv.function = GXxor;
|
|
gcv.line_style = LineOnOffDash;
|
|
/* gcv.line_width = iSelectionBorderThickness; */
|
|
gc = fe_GetGC(CONTEXT_WIDGET(context), gc_flags, &gcv);
|
|
XDrawRectangle(fe_display,
|
|
XtWindow(CONTEXT_DATA(context)->drawing_area), gc,
|
|
pRect->left, pRect->top,
|
|
pRect->right - pRect->left, pRect->bottom - pRect->top);
|
|
}
|
|
|
|
void
|
|
FE_DisplayEntireTableOrCell(MWContext* context, LO_Element* pLoElement)
|
|
{
|
|
LO_TableStruct* table;
|
|
|
|
if (pLoElement->lo_any.type == LO_TABLE)
|
|
{
|
|
XFE_DisplayTable(context, 0, &(pLoElement->lo_table));
|
|
}
|
|
else if (pLoElement->lo_any.type == LO_CELL)
|
|
{
|
|
XFE_DisplayCell(context, 0, &(pLoElement->lo_cell));
|
|
}
|
|
else /* shouldn't happen */
|
|
{
|
|
#ifdef DEBUG_akkana
|
|
printf("FE_DisplayEntireTableOrCell() with type %d\n",
|
|
pLoElement->lo_any.type);
|
|
#endif /* DEBUG_akkana */
|
|
table = lo_GetParentTable(context, pLoElement);
|
|
XFE_DisplayTable(context, 0, table);
|
|
}
|
|
}
|
|
|
|
|
|
#endif /* EDITOR */
|