-/* This file is part of
- * ======================================================
- *
- * LyX, The Document Processor
- *
- * Copyright 2000-2001 The LyX Team.
+/**
+ * \file FormGraphics.C
+ * This file is part of LyX, the document processor.
+ * Licence details can be found in the file COPYING.
*
- * ======================================================
+ * \author Baruch Even
+ * \author Herbert Voß
+ * \author Rob Lahaye
*
- * \file FormGraphics.C
- * \author Baruch Even, baruch.even@writeme.com
+ * Full author contact details are available in file CREDITS.
*/
-#include <config.h>
+#include <config.h>
-#ifdef __GNUG__
-#pragma implementation
-#endif
-
-#include "xformsBC.h"
-#include "ControlGraphics.h"
#include "FormGraphics.h"
-#include "form_graphics.h"
+#include "ControlGraphics.h"
+#include "forms/form_graphics.h"
-#include "xforms_helpers.h"
+#include "controllers/helper_funcs.h"
+
+#include "checkedwidgets.h"
#include "input_validators.h"
-#include "debug.h" // for lyxerr
-#include "support/lstrings.h" // for strToDbl & tostr
-#include "support/FileInfo.h"
-#include "support/filetools.h" // for MakeAbsPath etc
-#include "insets/insetgraphicsParams.h"
+#include "Tooltips.h"
+#include "xforms_helpers.h"
+#include "xformsBC.h"
+
#include "lyxrc.h" // for lyxrc.display_graphics
+#include "insets/insetgraphicsParams.h"
+
+#include "frontends/Alert.h"
+
+#include "support/lyxlib.h" // for float_equal
+#include "support/convert.h"
+
+#include "lyx_forms.h"
+
+#include <cmath>
+
+#ifndef CXX_GLOBAL_CSTD
+using std::floor;
+#endif
+
using std::endl;
+using std::vector;
+using std::string;
+
+namespace lyx {
+
+using support::bformat;
+using support::float_equal;
+using support::getStringFromVector;
+using support::token;
+
+namespace frontend {
+
namespace {
// Bound the number of input characters
-int const SCALE_MAXDIGITS = 3;
-int const WIDTH_MAXDIGITS = 3;
-int const HEIGHT_MAXDIGITS = 3;
-int const ROTATE_MAXCHARS = 4;
+int const SIZE_MAXDIGITS = 10;
int const FILENAME_MAXCHARS = 1024;
-
+
+string defaultUnit("cm");
+
+#if FL_VERSION == 0 || (FL_REVISION == 0 && FL_FIXLEVEL < 2)
+bool const scalableTabfolders = false;
+#else
+bool const scalableTabfolders = true;
+#endif
+
} // namespace anon
-typedef FormCB<ControlGraphics, FormDB<FD_form_graphics> > base_class;
+typedef FormController<ControlGraphics, FormView<FD_graphics> > base_class;
-FormGraphics::FormGraphics(ControlGraphics & c)
- : base_class(c, _("Graphics"))
+FormGraphics::FormGraphics(Dialog & parent)
+ : base_class(parent, _("Graphics"), scalableTabfolders)
{}
void FormGraphics::redraw()
{
- if(form() && form()->visible)
+ if (form() && form()->visible)
fl_redraw_form(form());
else
return;
-
- FL_FORM * outer_form = fl_get_active_folder(dialog_->tabFolder);
+ FL_FORM * outer_form = fl_get_active_folder(dialog_->tabfolder);
if (outer_form && outer_form->visible)
fl_redraw_form(outer_form);
}
void FormGraphics::build()
{
- dialog_.reset(build_graphics());
+ dialog_.reset(build_graphics(this));
+
+ // Manage the ok, apply, restore and cancel/close buttons
+ bcview().setOK(dialog_->button_ok);
+ bcview().setApply(dialog_->button_apply);
+ bcview().setCancel(dialog_->button_close);
+ bcview().setRestore(dialog_->button_restore);
+
+ // The file section.
+ file_.reset(build_graphics_file(this));
+
+ // Disable for read-only documents.
+ bcview().addReadOnly(file_->button_browse);
+ bcview().addReadOnly(file_->check_aspectratio);
+ bcview().addReadOnly(file_->check_draft);
+ bcview().addReadOnly(file_->check_nounzip);
+
+ // Check validity of "length + unit" input.
+ addCheckedGlueLength(bcview(), file_->input_width);
+ addCheckedGlueLength(bcview(), file_->input_height);
+
+ // Trigger an input event for cut&paste with middle mouse button.
+ setPrehandler(file_->input_filename);
+ setPrehandler(file_->input_lyxscale);
+ setPrehandler(file_->input_width);
+ setPrehandler(file_->input_height);
+
+ // Activate ok/apply immediately upon input.
+ fl_set_input_return(file_->input_filename, FL_RETURN_CHANGED);
+ fl_set_input_return(file_->input_lyxscale, FL_RETURN_CHANGED);
+ fl_set_input_return(file_->input_width, FL_RETURN_CHANGED);
+ fl_set_input_return(file_->input_height, FL_RETURN_CHANGED);
+
+ fl_set_input_maxchars(file_->input_filename, FILENAME_MAXCHARS);
+ fl_set_input_filter(file_->input_lyxscale, fl_unsigned_int_filter);
+
+ // Width default is scaling: use unsigned float filter.
+ fl_set_input_filter(file_->input_width, fl_unsigned_float_filter);
+ fl_set_input_maxchars(file_->input_height, SIZE_MAXDIGITS);
+
+ string const display_List =
+ _("Default|Monochrome|Grayscale|Color|Do not display");
+ fl_addto_choice(file_->choice_display, display_List.c_str());
+
+ string const height_list = buildChoiceLengthString();
+ string const width_list = bformat(_("Scale%%%%|%1$s"), height_list);
+
+ fl_addto_choice(file_->choice_width, width_list.c_str());
+ fl_addto_choice(file_->choice_height, height_list.c_str());
+
+ // set up the tooltips for the filesection
+ string str = _("The file you want to insert.");
+ tooltips().init(file_->input_filename, str);
+ str = _("Browse the directories.");
+ tooltips().init(file_->button_browse, str);
+
+ str = _("Scale the image to inserted percentage value.");
+ tooltips().init(file_->input_lyxscale, str);
+ str = _("Select display mode for this image.");
+ tooltips().init(file_->choice_display, str);
+
+ str = _("Set the image width to the inserted value.");
+ tooltips().init(file_->input_width, str);
+ // xgettext:no-c-format
+ str = _("Select unit for width; Scale% for scaling whole image.");
+ tooltips().init(file_->choice_width, str);
+ str = _("Set the image height to the inserted value.");
+ tooltips().init(file_->input_height, str);
+ str = _("Select unit for height.");
+ tooltips().init(file_->choice_height, str);
+ str = _("Do not distort the image. "
+ "Keep image within \"width\" by \"height\" and obey "
+ "aspect ratio.");
+ tooltips().init(file_->check_aspectratio, str);
+ str = _("Pass a filename like \"file.eps.gz\" to the LaTeX output. "
+ "Useful when LaTeX should unzip the file. Needs an additional "
+ "file like \"file.eps.bb\" which holds the values for "
+ "the bounding box.");
+ tooltips().init(file_->check_nounzip, str);
+ str = _("Show image only as a rectangle of the original size.");
+ tooltips().init(file_->check_draft, str);
- // Manage the ok, apply, restore and cancel/close buttons
- bc().setOK(dialog_->button_ok);
-// bc().setApply(dialog_->button_apply);
- bc().setCancel(dialog_->button_cancel);
- bc().setRestore(dialog_->button_restore);
-
- // the file section
- file_.reset(build_file());
+ // the bounding box selection
+ bbox_.reset(build_graphics_bbox(this));
+
+ // disable for read-only documents
+ bcview().addReadOnly(bbox_->button_getBB);
+ bcview().addReadOnly(bbox_->check_clip);
+
+ // check validity of "length + unit" input
+ addCheckedLyXLength(bcview(), bbox_->input_bb_x1, bbox_->text_X);
+
+ // trigger an input event for cut&paste with middle mouse button.
+ setPrehandler(bbox_->input_bb_x0);
+ setPrehandler(bbox_->input_bb_y0);
+ setPrehandler(bbox_->input_bb_x1);
+ setPrehandler(bbox_->input_bb_y1);
+
+ // for activate ok/apply immediately upon input
+ fl_set_input_return(bbox_->input_bb_x0, FL_RETURN_CHANGED);
+ fl_set_input_return(bbox_->input_bb_y0, FL_RETURN_CHANGED);
+ fl_set_input_return(bbox_->input_bb_x1, FL_RETURN_CHANGED);
+ fl_set_input_return(bbox_->input_bb_y1, FL_RETURN_CHANGED);
+
+ fl_set_input_filter(bbox_->input_bb_x0, fl_unsigned_float_filter);
+ fl_set_input_filter(bbox_->input_bb_y0, fl_unsigned_float_filter);
+ fl_set_input_filter(bbox_->input_bb_y1, fl_unsigned_float_filter);
+
+ string const bb_units = getStringFromVector(getBBUnits(), "|");
+ fl_addto_choice(bbox_->choice_bb_units, bb_units.c_str());
+
+ // set up the tooltips for the bounding-box-section
+ str = _("The lower left x-value of the bounding box.");
+ tooltips().init(bbox_->input_bb_x0, str);
+ str = _("The lower left y-value of the bounding box.");
+ tooltips().init(bbox_->input_bb_y0, str);
+ str = _("The upper right x-value of the bounding box; "
+ "only this input field allows length + unit, e.g. 5cm "
+ "and sets the unit for the other input fields.");
+ tooltips().init(bbox_->input_bb_x1, str);
+ str = _("The upper right y-value of the bounding box.");
+ tooltips().init(bbox_->input_bb_y1, str);
+ str = _("Select unit for the bounding box values.");
+ tooltips().init(bbox_->choice_bb_units, str);
+
+ str = _("Read the image coordinates new from file. For (e)ps-file "
+ "the bounding box is read, otherwise the imagesize in pixels. "
+ "Default unit is \"bp\", the PostScript's b(ig) p(oint).");
+ tooltips().init(bbox_->button_getBB, str);
+
+ str = _("Clip image to the bounding box values.");
+ tooltips().init(bbox_->check_clip, str);
+
+ // the extra section
+ extra_.reset(build_graphics_extra(this));
+
+ // disable for read-only documents
+ bcview().addReadOnly(extra_->input_rotate_angle);
+ bcview().addReadOnly(extra_->choice_origin);
+ bcview().addReadOnly(extra_->check_subcaption);
+ bcview().addReadOnly(extra_->input_special);
+
+ // trigger an input event for cut&paste with middle mouse button.
+ setPrehandler(extra_->input_rotate_angle);
+ setPrehandler(extra_->input_subcaption);
+ setPrehandler(extra_->input_special);
+
+ fl_set_input_return(extra_->input_rotate_angle, FL_RETURN_CHANGED);
+ fl_set_input_return(extra_->input_subcaption, FL_RETURN_CHANGED);
+ fl_set_input_return(extra_->input_special, FL_RETURN_CHANGED);
+
+ fl_set_input_filter(extra_->input_rotate_angle, fl_float_filter);
+
+ vector<RotationOriginPair> origindata = getRotationOriginData();
+
+ // Store the identifiers for later
+ origins_ = getSecond(origindata);
+
+ string const choice = getStringFromVector(getFirst(origindata), "|");
+ fl_addto_choice(extra_->choice_origin, choice.c_str());
+
+ // set up the tooltips for the extra section
+ str = _("Insert the rotation angle in degrees. "
+ "Positive value rotates anti-clockwise, "
+ "negative value clockwise.");
+ tooltips().init(extra_->input_rotate_angle, str);
+ str = _("Insert the point of origin for rotation.");
+ tooltips().init(extra_->choice_origin, str);
+ str = _("Enables use of subfigure with its own caption.");
+ tooltips().init(extra_->check_subcaption, str);
+ str = _("Insert the optional subfigure caption.");
+ tooltips().init(extra_->input_subcaption, str);
+ str = _("Add any additional LaTeX option, which is defined in the "
+ "graphicx-package and not mentioned in the gui's tabfolders.");
+ tooltips().init(extra_->input_special, str);
+
+ // Enable the tabfolder to be rescaled correctly.
+ if (scalableTabfolders)
+ fl_set_tabfolder_autofit(dialog_->tabfolder, FL_FIT);
+
+ // Stack tabs
+ fl_addto_tabfolder(dialog_->tabfolder, _("File").c_str(),
+ file_->form);
+ fl_addto_tabfolder(dialog_->tabfolder, _("Bounding Box").c_str(),
+ bbox_->form);
+ fl_addto_tabfolder(dialog_->tabfolder, _("Extra").c_str(),
+ extra_->form);
+
+ // set the right default unit
+ switch (lyxrc.default_papersize) {
+ case PAPER_DEFAULT: break;
+ case PAPER_USLETTER:
+ case PAPER_LEGALPAPER:
+ case PAPER_EXECUTIVEPAPER: defaultUnit = "in"; break;
+ case PAPER_A3PAPER:
+ case PAPER_A4PAPER:
+ case PAPER_A5PAPER:
+ case PAPER_B5PAPER: defaultUnit = "cm"; break;
+ }
+}
- fl_set_input_return (file_->input_filename, FL_RETURN_CHANGED);
- fl_set_input_return (file_->input_subcaption, FL_RETURN_CHANGED);
- fl_set_input_return (file_->input_lyxwidth, FL_RETURN_CHANGED);
- fl_set_input_return (file_->input_lyxheight, FL_RETURN_CHANGED);
- fl_set_input_maxchars(file_->input_filename, FILENAME_MAXCHARS);
- fl_addto_choice(file_->choice_width_lyxwidth, choice_Length_All.c_str());
- fl_addto_choice(file_->choice_width_lyxheight, choice_Length_All.c_str());
- fl_addto_choice(file_->choice_display,
- _(" Default | Monochrome | Grayscale | Color | Do not display "));
- //FIXME: This does not work (always set to monochrome by default. Why? JSpitzm)
- fl_set_choice(file_->choice_display, 1);
+void FormGraphics::apply()
+{
+ // Create the parameters structure and fill the data from the dialog.
+ InsetGraphicsParams & igp = controller().params();
- bc().addReadOnly(file_->button_browse);
- bc().addReadOnly(file_->check_subcaption);
- bc().addReadOnly(file_->choice_display);
+ // the file section
+ igp.filename.set(getString(file_->input_filename),
+ kernel().bufferFilepath());
- // the size section
- size_.reset(build_size());
+ igp.lyxscale = convert<int>(getString(file_->input_lyxscale));
+ if (igp.lyxscale == 0) {
+ igp.lyxscale = 100;
+ }
- fl_set_input_return (size_->input_scale, FL_RETURN_CHANGED);
- fl_set_input_return (size_->input_width, FL_RETURN_CHANGED);
- fl_set_input_return (size_->input_height, FL_RETURN_CHANGED);
+ switch (fl_get_choice(file_->choice_display)) {
+ case 5:
+ igp.display = graphics::NoDisplay;
+ break;
+ case 4:
+ igp.display = graphics::ColorDisplay;
+ break;
+ case 3:
+ igp.display = graphics::GrayscaleDisplay;
+ break;
+ case 2:
+ igp.display = graphics::MonochromeDisplay;
+ break;
+ case 1:
+ igp.display = graphics::DefaultDisplay;
+ }
- fl_set_input_maxchars(size_->input_scale, SCALE_MAXDIGITS);
- fl_set_input_maxchars(size_->input_width, WIDTH_MAXDIGITS);
- fl_set_input_maxchars(size_->input_height, HEIGHT_MAXDIGITS);
+ // first item in choice_width means scaling
+ if (fl_get_choice(file_->choice_width) == 1) {
+ igp.scale = getString(file_->input_width);
+ if (igp.scale.empty()
+ || float_equal(convert<double>(igp.scale), 0.0, 0.05)
+ || float_equal(convert<double>(igp.scale), 100.0, 0.05)) {
+ igp.scale.erase();
+ }
+ igp.width = LyXLength();
+ } else {
+ igp.scale.erase();
+ igp.width = LyXLength(getLengthFromWidgets(file_->input_width,
+ file_->choice_width));
+ }
+ igp.height = LyXLength(getLengthFromWidgets(file_->input_height,
+ file_->choice_height));
+ igp.keepAspectRatio = fl_get_button(file_->check_aspectratio);
+
+ igp.draft = fl_get_button(file_->check_draft);
+ igp.noUnzip = fl_get_button(file_->check_nounzip);
+
+ // the bb section
+ if (!controller().bbChanged) {
+ // don't write anything
+ igp.bb.erase();
+ } else {
+ // allow length + unit input only for x1 input field
+ string x1_str = "0";
+ if (!getString(bbox_->input_bb_x1).empty()) {
+ x1_str = getLengthFromWidgets(bbox_->input_bb_x1,
+ bbox_->choice_bb_units);
+ LyXLength x1 = LyXLength(x1_str);
+ x1_str = x1.asString();
+
+ string unit;
+ switch (x1.unit()) {
+ case LyXLength::IN:
+ unit = "in";
+ break;
+ case LyXLength::MM:
+ unit = "mm";
+ break;
+ case LyXLength::CM:
+ unit = "cm";
+ break;
+ case LyXLength::BP:
+ default:
+ unit = "bp";
+ }
+ fl_set_choice_text(bbox_->choice_bb_units, unit.c_str());
+ }
- fl_set_input_filter(size_->input_scale, fl_unsigned_float_filter);
+ string bb;
+ if (getString(bbox_->input_bb_x0).empty())
+ bb = "0";
+ else
+ bb = getLengthFromWidgets(bbox_->input_bb_x0,
+ bbox_->choice_bb_units);
- fl_addto_choice(size_->choice_width_units, choice_Length_All.c_str());
- fl_addto_choice(size_->choice_height_units, choice_Length_All.c_str());
+ bb += ' ';
- bc().addReadOnly(size_->button_default);
- bc().addReadOnly(size_->button_wh);
- bc().addReadOnly(size_->button_scale);
- bc().addReadOnly(size_->check_aspectratio);
+ if (getString(bbox_->input_bb_y0).empty())
+ bb += '0';
+ else
+ bb += getLengthFromWidgets(bbox_->input_bb_y0,
+ bbox_->choice_bb_units);
- // the bounding box selection
- bbox_.reset(build_bbox());
- fl_set_input_return (bbox_->input_bb_x0, FL_RETURN_CHANGED);
- fl_set_input_return (bbox_->input_bb_y0, FL_RETURN_CHANGED);
- fl_set_input_return (bbox_->input_bb_x1, FL_RETURN_CHANGED);
- fl_set_input_return (bbox_->input_bb_y1, FL_RETURN_CHANGED);
-
- bc().addReadOnly(bbox_->button_getBB);
- bc().addReadOnly(bbox_->button_clip);
- bc().addReadOnly(bbox_->button_draft);
-
- string const bb_units = "pt|cm|in";
- fl_addto_choice(bbox_->choice_bb_x0, bb_units.c_str());
- fl_addto_choice(bbox_->choice_bb_y0, bb_units.c_str());
- fl_addto_choice(bbox_->choice_bb_x1, bb_units.c_str());
- fl_addto_choice(bbox_->choice_bb_y1, bb_units.c_str());
-
-
- // the rotate section
- special_.reset(build_special());
-
- fl_set_input_return (special_->input_rotate_angle, FL_RETURN_CHANGED);
-
- string const choice_origin =
- "default|" // not important
- "leftTop|leftBottom|leftBaseline|" // lt lb lB
- "center|" // c
- "centerTop|centerBottom|centerBaseline|" // ct cb cB
- "rightTop|rightBottom|rightBaseline|" // rt rb rB
- "referencePoint"; // special
- fl_addto_choice(special_->choice_origin, choice_origin.c_str());
-
- fl_set_input_return (special_->input_special, FL_RETURN_CHANGED);
- fl_set_input_maxchars(special_->input_rotate_angle, ROTATE_MAXCHARS);
- fl_set_input_filter(special_->input_rotate_angle, fl_float_filter);
-
- // add the different tabfolders
- fl_addto_tabfolder(dialog_->tabFolder, _("File"), file_->form);
- fl_addto_tabfolder(dialog_->tabFolder, _("Size"), size_->form);
- fl_addto_tabfolder(dialog_->tabFolder, _("Bounding Box"), bbox_->form);
- fl_addto_tabfolder(dialog_->tabFolder, _("Extras"), special_->form);
-}
+ bb += ' ' + x1_str + ' ';
+ if (getString(bbox_->input_bb_y1).empty())
+ bb += '0';
+ else
+ bb += getLengthFromWidgets(bbox_->input_bb_y1,
+ bbox_->choice_bb_units);
-void FormGraphics::apply()
-{
- // Create the parameters structure and fill the data from the dialog.
- InsetGraphicsParams & igp = controller().params();
- igp.filename = getStringFromInput(file_->input_filename);
- if (!controller().bbChanged) // different to the original one?
- igp.bb = string(); // don't write anything
- else {
- string bb;
- if (getStringFromInput(bbox_->input_bb_x0).empty())
- bb = "0 ";
- else
- bb = getLengthFromWidgets(bbox_->input_bb_x0,
- bbox_->choice_bb_x0)+" ";
- if (getStringFromInput(bbox_->input_bb_y0).empty())
- bb += "0 ";
- else
- bb += (getLengthFromWidgets(bbox_->input_bb_y0,
- bbox_->choice_bb_y0)+" ");
- if (getStringFromInput(bbox_->input_bb_x1).empty())
- bb += "0 ";
- else
- bb += (getLengthFromWidgets(bbox_->input_bb_x1,
- bbox_->choice_bb_x1)+" ");
- if (getStringFromInput(bbox_->input_bb_y1).empty())
- bb += "0 ";
- else
- bb += (getLengthFromWidgets(bbox_->input_bb_y1,
- bbox_->choice_bb_y1)+" ");
- igp.bb = bb;
+ igp.bb = bb;
}
- igp.draft = fl_get_button(bbox_->button_draft);
- igp.clip = fl_get_button(bbox_->button_clip);
- igp.subcaption = fl_get_button(file_->check_subcaption);
- igp.subcaptionText = getStringFromInput(file_->input_subcaption);
- // use preferences settings if choice is set to default
- if (fl_get_choice(file_->choice_display) == 1) {
- if (lyxrc.display_graphics == "mono")
- igp.display = InsetGraphicsParams::MONOCHROME;
- else if (lyxrc.display_graphics == "gray")
- igp.display = InsetGraphicsParams::GRAYSCALE;
- else if (lyxrc.display_graphics == "color")
- igp.display = InsetGraphicsParams::COLOR;
- else if (lyxrc.display_graphics == "no")
- igp.display = InsetGraphicsParams::NONE;
- } else if (fl_get_choice(file_->choice_display) == 2) {
- igp.display = InsetGraphicsParams::MONOCHROME;
- } else if (fl_get_choice(file_->choice_display) == 3) {
- igp.display = InsetGraphicsParams::GRAYSCALE;
- } else if (fl_get_choice(file_->choice_display) == 4) {
- igp.display = InsetGraphicsParams::COLOR;
- } else if (fl_get_choice(file_->choice_display) == 5) {
- igp.display = InsetGraphicsParams::NONE;
+ igp.clip = fl_get_button(bbox_->check_clip);
+
+ // the extra section
+ igp.rotateAngle = getString(extra_->input_rotate_angle);
+
+ // map angle into -360 (clock-wise) to +360 (counter clock-wise)
+ double rotAngle = convert<double>(igp.rotateAngle);
+ if (std::abs(rotAngle) > 360.0) {
+ rotAngle -= 360.0 * floor(rotAngle / 360.0);
+ igp.rotateAngle = convert<string>(rotAngle);
}
- if (fl_get_button(size_->button_default))
- igp.size_type = InsetGraphicsParams::DEFAULT_SIZE;
- else if (fl_get_button(size_->button_wh))
- igp.size_type = InsetGraphicsParams::WH;
- else
- igp.size_type = InsetGraphicsParams::SCALE;
- igp.width = LyXLength(getLengthFromWidgets(size_->input_width,
- size_->choice_width_units));
- igp.height = LyXLength(getLengthFromWidgets(size_->input_height,
- size_->choice_height_units));
- igp.scale = strToInt(getStringFromInput(size_->input_scale));
- igp.keepAspectRatio = fl_get_button(size_->check_aspectratio);
- igp.lyxwidth = LyXLength(getLengthFromWidgets(file_->input_lyxwidth,
- file_->choice_width_lyxwidth));
- igp.lyxheight = LyXLength(getLengthFromWidgets(file_->input_lyxheight,
- file_->choice_width_lyxheight));
-
- igp.rotateAngle =
- strToDbl(getStringFromInput(special_->input_rotate_angle));
- while (igp.rotateAngle < 0.0 || igp.rotateAngle > 360.0) {
- if (igp.rotateAngle < 0.0) {
- igp.rotateAngle += 360.0;
- } else if (igp.rotateAngle > 360.0) {
- igp.rotateAngle -= 360.0;
- }
+
+ fl_set_input(extra_->input_rotate_angle, igp.rotateAngle.c_str());
+
+ int const origin_pos = fl_get_choice(extra_->choice_origin);
+ if (origin_pos == 0) {
+ igp.rotateOrigin.erase();
+ } else {
+ igp.rotateOrigin = origins_[origin_pos - 1];
}
- if (fl_get_choice(special_->choice_origin) > 0)
- igp.rotateOrigin = fl_get_choice_text(special_->choice_origin);
- else
- igp.rotateOrigin = string();
- igp.special = getStringFromInput(special_->input_special);
- igp.testInvariant();
+
+ igp.subcaption = fl_get_button(extra_->check_subcaption);
+ igp.subcaptionText = getString(extra_->input_subcaption);
+
+ igp.special = getString(extra_->input_special);
}
-void FormGraphics::update()
-{ string unit = "cm";
- if (lyxrc.default_papersize < 3)
- unit = "in";
- string const defaultUnit = string(unit);
+void FormGraphics::update() {
// Update dialog with details from inset
InsetGraphicsParams & igp = controller().params();
- fl_set_input(file_->input_filename, igp.filename.c_str());
- // set the bounding box values, if exists. First we need the whole
- // path, because the controller knows nothing about the doc-dir
- lyxerr << "GraphicsUpdate::BoundingBox = " << igp.bb << "\n";
- controller().bbChanged = false;
- if (igp.bb.empty()) {
- string const fileWithAbsPath = MakeAbsPath(igp.filename, OnlyPath(igp.filename));
- string bb = controller().readBB(fileWithAbsPath);
- lyxerr << "file::BoundingBox = " << bb << "\n";
- if (!bb.empty()) {
- // get the values from the file
- // in this case we always have the point-unit
- fl_set_input(bbox_->input_bb_x0, token(bb,' ',0).c_str());
- fl_set_input(bbox_->input_bb_y0, token(bb,' ',1).c_str());
- fl_set_input(bbox_->input_bb_x1, token(bb,' ',2).c_str());
- fl_set_input(bbox_->input_bb_y1, token(bb,' ',3).c_str());
- }
- } else { // get the values from the inset
- controller().bbChanged = true;
- LyXLength anyLength;
- anyLength = LyXLength(token(igp.bb,' ',0));
- updateWidgetsFromLength(bbox_->input_bb_x0,
- bbox_->choice_bb_x0,anyLength,"pt");
- anyLength = LyXLength(token(igp.bb,' ',1));
- updateWidgetsFromLength(bbox_->input_bb_y0,
- bbox_->choice_bb_y0,anyLength,"pt");
- anyLength = LyXLength(token(igp.bb,' ',2));
- updateWidgetsFromLength(bbox_->input_bb_x1,
- bbox_->choice_bb_x1,anyLength,"pt");
- anyLength = LyXLength(token(igp.bb,' ',3));
- updateWidgetsFromLength(bbox_->input_bb_y1,
- bbox_->choice_bb_y1,anyLength,"pt");
- }
- // Update the draft and clip mode
- fl_set_button(bbox_->button_draft, igp.draft);
- fl_set_button(bbox_->button_clip, igp.clip);
- // Update the subcaption check button and input field
- fl_set_button(file_->check_subcaption, igp.subcaption);
- fl_set_input(file_->input_subcaption, igp.subcaptionText.c_str());
- setEnabled(file_->input_subcaption,
- fl_get_button(file_->check_subcaption));
+
+ // the file section
+ string const name =
+ igp.filename.outputFilename(kernel().bufferFilepath());
+ fl_set_input(file_->input_filename, name.c_str());
+ fl_set_input(file_->input_lyxscale, convert<string>(igp.lyxscale).c_str());
switch (igp.display) {
- case InsetGraphicsParams::NONE: { // dont't display
+ case graphics::NoDisplay:
fl_set_choice(file_->choice_display, 5);
break;
- }
- case InsetGraphicsParams::MONOCHROME: {
- fl_set_choice(file_->choice_display, 2);
+ case graphics::ColorDisplay:
+ fl_set_choice(file_->choice_display, 4);
break;
- }
- case InsetGraphicsParams::GRAYSCALE: {
+ case graphics::GrayscaleDisplay:
fl_set_choice(file_->choice_display, 3);
break;
- }
- case InsetGraphicsParams::COLOR: {
- fl_set_choice(file_->choice_display, 4);
+ case graphics::MonochromeDisplay:
+ fl_set_choice(file_->choice_display, 2);
break;
- }
+ case graphics::DefaultDisplay:
+ fl_set_choice(file_->choice_display, 1);
}
- updateWidgetsFromLength(size_->input_width,
- size_->choice_width_units,igp.width,defaultUnit);
- updateWidgetsFromLength(size_->input_height,
- size_->choice_height_units,igp.height,defaultUnit);
- fl_set_input(size_->input_scale, tostr(igp.scale).c_str());
- switch (igp.size_type) {
- case InsetGraphicsParams::DEFAULT_SIZE: {
- fl_set_button(size_->button_default,1);
- setEnabled(size_->input_width, 0);
- setEnabled(size_->choice_width_units, 0);
- setEnabled(size_->input_height, 0);
- setEnabled(size_->choice_height_units, 0);
- setEnabled(size_->input_scale, 0);
- break;
- }
- case InsetGraphicsParams::WH: {
- fl_set_button(size_->button_wh, 1);
- setEnabled(size_->input_width, 1);
- setEnabled(size_->choice_width_units, 1);
- setEnabled(size_->input_height, 1);
- setEnabled(size_->choice_height_units, 1);
- setEnabled(size_->input_scale, 0);
- break;
- }
- case InsetGraphicsParams::SCALE: {
- fl_set_button(size_->button_scale, 1);
- setEnabled(size_->input_width, 0);
- setEnabled(size_->choice_width_units, 0);
- setEnabled(size_->input_height, 0);
- setEnabled(size_->choice_height_units, 0);
- setEnabled(size_->input_scale, 1);
- break;
- }
+
+ // set width input fields according to scaling or width/height input
+ if (!igp.scale.empty() && !float_equal(convert<double>(igp.scale), 0.0, 0.05)) {
+ fl_set_input_filter(file_->input_width, fl_unsigned_float_filter);
+ fl_set_input_maxchars(file_->input_width, 0);
+ fl_set_input(file_->input_width, igp.scale.c_str());
+ fl_set_choice(file_->choice_width, 1);
+ } else {
+ fl_set_input_filter(file_->input_width, NULL);
+ fl_set_input_maxchars(file_->input_width, SIZE_MAXDIGITS);
+ updateWidgetsFromLength(file_->input_width, file_->choice_width,
+ igp.width, defaultUnit);
}
- fl_set_button(size_->check_aspectratio, igp.keepAspectRatio);
- // now the lyx-internally viewsize
- updateWidgetsFromLength(file_->input_lyxwidth,
- file_->choice_width_lyxwidth, igp.lyxwidth,defaultUnit);
- updateWidgetsFromLength(file_->input_lyxheight,
- file_->choice_width_lyxheight, igp.lyxheight,defaultUnit);
- // Update the rotate angle and special commands
- fl_set_input(special_->input_rotate_angle,
- tostr(igp.rotateAngle).c_str());
- if (igp.rotateOrigin.empty())
- fl_set_choice(special_->choice_origin,0);
- else
- fl_set_choice_text(special_->choice_origin,igp.rotateOrigin.c_str());
- fl_set_input(special_->input_special, igp.special.c_str());
-}
+ updateWidgetsFromLength(file_->input_height, file_->choice_height,
+ igp.height, defaultUnit);
+
+ // disable height input in case of scaling
+ bool const disable_height = (!igp.scale.empty()
+ && !float_equal(convert<double>(igp.scale), 0.0, 0.05));
+ setEnabled(file_->input_height, !disable_height);
+ setEnabled(file_->choice_height, !disable_height);
+
+ fl_set_button(file_->check_aspectratio, igp.keepAspectRatio);
+ fl_set_button(file_->check_draft, igp.draft);
+ fl_set_button(file_->check_nounzip, igp.noUnzip);
+
+ // disable aspectratio button in case of scaling or one of width/height
+ // is empty
+ bool const disable_aspectRatio = disable_height ||
+ getString(file_->input_width).empty() ||
+ getString(file_->input_height).empty();
+ setEnabled(file_->check_aspectratio, !disable_aspectRatio);
+
+ // the bb section
+ // set the bounding box values, if exists. First we need the whole
+ // path, because the controller knows nothing about the doc-dir
+ updateBB(igp.filename.absFilename(), igp.bb);
+ fl_set_button(bbox_->check_clip, igp.clip);
+
+ // the extra section
+ fl_set_input(extra_->input_rotate_angle, igp.rotateAngle.c_str());
+
+ int origin_pos = 1;
+ if (!igp.rotateOrigin.empty()) {
+ origin_pos += findPos(origins_, igp.rotateOrigin);
+ }
+ fl_set_choice(extra_->choice_origin, origin_pos);
+
+ fl_set_button(extra_->check_subcaption, igp.subcaption);
+ fl_set_input(extra_->input_subcaption, igp.subcaptionText.c_str());
+ setEnabled(extra_->input_subcaption,
+ fl_get_button(extra_->check_subcaption));
+ fl_set_input(extra_->input_special, igp.special.c_str());
+
+ // open dialog in the file-tab, whenever filename is empty
+ if (igp.filename.empty()) {
+ fl_set_folder(dialog_->tabfolder, file_->form);
+ }
+}
-namespace {
-bool isValid(FL_OBJECT * ob)
+void FormGraphics::updateBB(string const & filename, string const & bb_inset)
{
- string const input = getStringFromInput(ob);
- return input.empty() || isValidLength(input) || isStrDbl(input);
+ // Update dialog with details from inset
+ // set the bounding box values, if exists. First we need the whole
+ // path, because the controller knows nothing about the doc-dir
+ controller().bbChanged = false;
+ if (bb_inset.empty()) {
+ lyxerr[Debug::GRAPHICS]
+ << "FormGraphics::updateBB() [no BoundingBox]" << endl;
+ string const bb = controller().readBB(filename);
+ if (!bb.empty()) {
+ // get the values from the file
+ // in this case we always have the point-unit
+ fl_set_input(bbox_->input_bb_x0,
+ token(bb,' ',0).c_str());
+ fl_set_input(bbox_->input_bb_y0,
+ token(bb,' ',1).c_str());
+ fl_set_input(bbox_->input_bb_x1,
+ token(bb,' ',2).c_str());
+ fl_set_input(bbox_->input_bb_y1,
+ token(bb,' ',3).c_str());
+
+ } else {
+ // no bb from file
+ fl_set_input(bbox_->input_bb_x0, bb.c_str());
+ fl_set_input(bbox_->input_bb_y0, bb.c_str());
+ fl_set_input(bbox_->input_bb_x1, bb.c_str());
+ fl_set_input(bbox_->input_bb_y1, bb.c_str());
+ }
+ // "bp"
+ fl_set_choice(bbox_->choice_bb_units, 1);
+
+ } else {
+ // get the values from the inset
+ lyxerr[Debug::GRAPHICS]
+ << "FormGraphics::updateBB(): igp has BoundingBox"
+ << " ["<< bb_inset << "]" << endl;
+ controller().bbChanged = true;
+
+ LyXLength anyLength;
+ anyLength = LyXLength(token(bb_inset,' ',0));
+ updateWidgetsFromLength(bbox_->input_bb_x0,
+ bbox_->choice_bb_units,anyLength,"bp");
+ anyLength = LyXLength(token(bb_inset,' ',1));
+ updateWidgetsFromLength(bbox_->input_bb_y0,
+ bbox_->choice_bb_units,anyLength,"bp");
+ anyLength = LyXLength(token(bb_inset,' ',2));
+ updateWidgetsFromLength(bbox_->input_bb_x1,
+ bbox_->choice_bb_units,anyLength,"bp");
+ anyLength = LyXLength(token(bb_inset,' ',3));
+ updateWidgetsFromLength(bbox_->input_bb_y1,
+ bbox_->choice_bb_units,anyLength,"bp");
+ }
}
-} // namespace anon
-
-
ButtonPolicy::SMInput FormGraphics::input(FL_OBJECT * ob, long)
{
+ ButtonPolicy::SMInput activate = ButtonPolicy::SMI_VALID;
+
+ // the file section
if (ob == file_->button_browse) {
+ activate = ButtonPolicy::SMI_NOOP;
+
// Get the filename from the dialog
- string const in_name = getStringFromInput(file_->input_filename);
- string const out_name = controller().Browse(in_name);
+ string const in_name = getString(file_->input_filename);
+ string const out_name = controller().browse(in_name);
+ lyxerr[Debug::GRAPHICS]
+ << "[FormGraphics]out_name: " << out_name << endl;
if (out_name != in_name && !out_name.empty()) {
fl_set_input(file_->input_filename, out_name.c_str());
}
- } else if (!controller().bbChanged &&
- ((ob == bbox_->input_bb_x0) || (ob == bbox_->input_bb_y0) ||
- (ob == bbox_->input_bb_x1) || (ob == bbox_->input_bb_y1) ||
- (ob == bbox_->choice_bb_x0) || (ob == bbox_->choice_bb_y0) ||
- (ob == bbox_->choice_bb_x1) || (ob == bbox_->choice_bb_y1))) {
- controller().bbChanged = true;
- } else if (ob == size_->button_default) {
- setEnabled(size_->input_width, 0);
- setEnabled(size_->choice_width_units, 0);
- setEnabled(size_->input_height, 0);
- setEnabled(size_->choice_height_units, 0);
- setEnabled(size_->input_scale, 0);
- } else if (ob == size_->button_wh) {
- setEnabled(size_->input_width, 1);
- setEnabled(size_->choice_width_units, 1);
- setEnabled(size_->input_height, 1);
- setEnabled(size_->choice_height_units, 1);
- setEnabled(size_->input_scale, 0);
- } else if (ob == size_->button_scale) {
- setEnabled(size_->input_width, 0);
- setEnabled(size_->choice_width_units, 0);
- setEnabled(size_->input_height, 0);
- setEnabled(size_->choice_height_units, 0);
- setEnabled(size_->input_scale, 1);
- } else if (ob == file_->check_subcaption) {
- setEnabled(file_->input_subcaption,
- fl_get_button(file_->check_subcaption));
- } else if (ob == bbox_->button_getBB) {
- string const filename = getStringFromInput(file_->input_filename);
- if (!filename.empty()) {
- string const fileWithAbsPath = MakeAbsPath(filename, OnlyPath(filename));
- string bb = controller().readBB(fileWithAbsPath);
- lyxerr << "getBB::BoundingBox = " << bb << "\n";
- if (!bb.empty()) {
- fl_set_input(bbox_->input_bb_x0, token(bb,' ',0).c_str());
- fl_set_input(bbox_->input_bb_y0, token(bb,' ',1).c_str());
- fl_set_input(bbox_->input_bb_x1, token(bb,' ',2).c_str());
- fl_set_input(bbox_->input_bb_y1, token(bb,' ',3).c_str());
+ if (controller().isFilenameValid(out_name) &&
+ !controller().bbChanged) {
+ updateBB(out_name, string());
+ }
+ } else if (ob == file_->input_width || ob == file_->input_height) {
+ // disable aspectratio button in case of scaling or one of
+ // width/height is empty
+ bool const disable = fl_get_choice(file_->choice_width) == 1 ||
+ getString(file_->input_width).empty() ||
+ getString(file_->input_height).empty();
+ setEnabled(file_->check_aspectratio, !disable);
+ } else if (ob == file_->choice_width) {
+ // disable height input in case of scaling
+ bool const scaling = fl_get_choice(file_->choice_width) == 1;
+ setEnabled(file_->input_height, !scaling);
+ setEnabled(file_->choice_height, !scaling);
+
+ // allow only integer intput for scaling; float otherwise
+ if (scaling) {
+ fl_set_input_filter(file_->input_width, fl_unsigned_float_filter);
+ fl_set_input_maxchars(file_->input_width, 0);
+ } else {
+ fl_set_input_filter(file_->input_width, NULL);
+ fl_set_input_maxchars(file_->input_width, SIZE_MAXDIGITS);
}
- controller().bbChanged = false;
- }
- } else if (ob == dialog_->button_help) {
- controller().help();
- }
- // Put verifications that the dialog shows some sane values,
- // if not disallow clicking on ok/apply.
- // Possibly use a label in the bottom of the dialog to give the reason.
- ButtonPolicy::SMInput activate = ButtonPolicy::SMI_VALID;
- // check if the input is valid
- bool invalid = !isValid(bbox_->input_bb_x0);
- invalid = invalid || !isValid(bbox_->input_bb_x1);
- invalid = invalid || !isValid(bbox_->input_bb_y0);
- invalid = invalid || !isValid(bbox_->input_bb_y1);
- invalid = invalid || !isValid(size_->input_width);
- invalid = invalid || !isValid(size_->input_height);
- invalid = invalid || !isValid(file_->input_lyxwidth);
- invalid = invalid || !isValid(file_->input_lyxheight);
-
- // deactivate OK/ Apply buttons and
- // spit out warnings if invalid
- if (ob == bbox_->input_bb_x0
- || ob == bbox_->input_bb_x1
- || ob == bbox_->input_bb_y0
- || ob == bbox_->input_bb_y1
- || ob == size_->input_width
- || ob == size_->input_height
- || ob == file_->input_lyxwidth
- || ob == file_->input_lyxheight) {
- if (invalid) {
- fl_set_object_label(dialog_->text_warning,
- _("Warning: Invalid Length!"));
- fl_show_object(dialog_->text_warning);
- activate = ButtonPolicy::SMI_INVALID;
+ // disable aspectratio button in case of scaling or height
+ // input is empty
+ bool const disable_aspectratio =
+ scaling || getString(file_->input_height).empty();
+ setEnabled(file_->check_aspectratio, !disable_aspectratio);
+ // the bb section
+ } else if (!controller().bbChanged &&
+ (ob == bbox_->check_clip || ob == bbox_->choice_bb_units ||
+ ob == bbox_->input_bb_x0 || ob == bbox_->input_bb_y0 ||
+ ob == bbox_->input_bb_x1 || ob == bbox_->input_bb_y1)) {
+ controller().bbChanged = true;
+ } else if (ob == bbox_->button_getBB) {
+ string const filename = getString(file_->input_filename);
+ if (!filename.empty()) {
+ string bb = controller().readBB(filename);
+ if (!bb.empty()) {
+ fl_set_input(bbox_->input_bb_x0, token(bb,' ',0).c_str());
+ fl_set_input(bbox_->input_bb_y0, token(bb,' ',1).c_str());
+ fl_set_input(bbox_->input_bb_x1, token(bb,' ',2).c_str());
+ fl_set_input(bbox_->input_bb_y1, token(bb,' ',3).c_str());
+ fl_set_choice_text(bbox_->choice_bb_units, "bp");
+ }
+ controller().bbChanged = false;
} else {
- fl_hide_object(dialog_->text_warning);
- activate = ButtonPolicy::SMI_VALID;
+ fl_set_input(bbox_->input_bb_x0, "");
+ fl_set_input(bbox_->input_bb_y0, "");
+ fl_set_input(bbox_->input_bb_x1, "");
+ fl_set_input(bbox_->input_bb_y1, "");
+ fl_set_choice_text(bbox_->choice_bb_units, "bp");
}
- return activate;
+ // the extra section
+ } else if (ob == extra_->check_subcaption) {
+ setEnabled(extra_->input_subcaption,
+ fl_get_button(extra_->check_subcaption));
+
+ } else if (ob == file_->button_edit) {
+ activate = ButtonPolicy::SMI_NOOP;
+ controller().editGraphics();
}
- return checkInput();
-}
-ButtonPolicy::SMInput FormGraphics::checkInput()
-{
- // Put verifications that the dialog shows some sane values,
- // if not disallow clicking on ok/apply.
- // Possibly use a label in the bottom of the dialog to give the reason.
- ButtonPolicy::SMInput activate = ButtonPolicy::SMI_VALID;
- // We verify now that there is a filename, it exists, it's a file
- // and it's readable.
- string filename = getStringFromInput(file_->input_filename);
- FileInfo file(filename);
- if (filename.empty() || !file.isOK() || !file.exist()
- || !file.isRegular() || !file.readable()
- )
- activate = ButtonPolicy::SMI_INVALID;
-
return activate;
}
+} // namespace frontend
+} // namespace lyx