summaryrefslogtreecommitdiffstats
path: root/comm/mail/components/compose/content/dialogs/EdImageDialog.js
blob: 91e558cd50988897564c7992eae5b57e5732e689 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
/* This Source Code Form is subject to the terms of the Mozilla Public
 * License, v. 2.0. If a copy of the MPL was not distributed with this
 * file, You can obtain one at http://mozilla.org/MPL/2.0/. */

/*
 Note: We encourage non-empty alt text for images inserted into a page.
 When there's no alt text, we always write 'alt=""' as the attribute, since "alt" is a required attribute.
 We allow users to not have alt text by checking a "Don't use alterate text" radio button,
 and we don't accept spaces as valid alt text. A space used to be required to avoid the error message
 if user didn't enter alt text, but is unnecessary now that we no longer annoy the user
 with the error dialog if alt="" is present on an img element.
 We trim all spaces at the beginning and end of user's alt text
*/

/* import-globals-from ../editorUtilities.js */
/* import-globals-from EdDialogCommon.js */

var gInsertNewImage = true;
var gDoAltTextError = false;
var gConstrainOn = false;
// Note used in current version, but these are set correctly
//  and could be used to reset width and height used for constrain ratio
var gConstrainWidth = 0;
var gConstrainHeight = 0;
var imageElement;
var gImageMap = 0;
var gCanRemoveImageMap = false;
var gRemoveImageMap = false;
var gImageMapDisabled = false;
var gActualWidth = "";
var gActualHeight = "";
var gOriginalSrc = "";
var gTimerID;
var gValidateTab;
var gInsertNewIMap;

// These must correspond to values in EditorDialog.css for each theme
// (unfortunately, setting "style" attribute here doesn't work!)
var gPreviewImageWidth = 80;
var gPreviewImageHeight = 50;

// dialog initialization code

function ImageStartup() {
  gDialog.tabBox = document.getElementById("TabBox");
  gDialog.tabLocation = document.getElementById("imageLocationTab");
  gDialog.tabDimensions = document.getElementById("imageDimensionsTab");
  gDialog.tabBorder = document.getElementById("imageBorderTab");
  gDialog.srcInput = document.getElementById("srcInput");
  gDialog.titleInput = document.getElementById("titleInput");
  gDialog.altTextInput = document.getElementById("altTextInput");
  gDialog.altTextRadioGroup = document.getElementById("altTextRadioGroup");
  gDialog.altTextRadio = document.getElementById("altTextRadio");
  gDialog.noAltTextRadio = document.getElementById("noAltTextRadio");
  gDialog.actualSizeRadio = document.getElementById("actualSizeRadio");
  gDialog.constrainCheckbox = document.getElementById("constrainCheckbox");
  gDialog.widthInput = document.getElementById("widthInput");
  gDialog.heightInput = document.getElementById("heightInput");
  gDialog.widthUnitsMenulist = document.getElementById("widthUnitsMenulist");
  gDialog.heightUnitsMenulist = document.getElementById("heightUnitsMenulist");
  gDialog.imagelrInput = document.getElementById("imageleftrightInput");
  gDialog.imagetbInput = document.getElementById("imagetopbottomInput");
  gDialog.border = document.getElementById("border");
  gDialog.alignTypeSelect = document.getElementById("alignTypeSelect");
  gDialog.PreviewWidth = document.getElementById("PreviewWidth");
  gDialog.PreviewHeight = document.getElementById("PreviewHeight");
  gDialog.PreviewImage = document.getElementById("preview-image");
  gDialog.PreviewImage.addEventListener("load", PreviewImageLoaded);
  gDialog.OkButton = document.querySelector("dialog").getButton("accept");
}

// Set dialog widgets with attribute data
// We get them from globalElement copy so this can be used
//   by AdvancedEdit(), which is shared by all property dialogs
function InitImage() {
  // Set the controls to the image's attributes
  var src = globalElement.getAttribute("src");

  // For image insertion the 'src' attribute is null.
  if (src) {
    // Shorten data URIs for display.
    shortenImageData(src, gDialog.srcInput);
  }

  // Set "Relativize" checkbox according to current URL state
  SetRelativeCheckbox();

  // Force loading of image from its source and show preview image
  LoadPreviewImage();

  gDialog.titleInput.value = globalElement.getAttribute("title");

  var hasAltText = globalElement.hasAttribute("alt");
  var altText = globalElement.getAttribute("alt");
  gDialog.altTextInput.value = altText;
  if (altText || (!hasAltText && globalElement.hasAttribute("src"))) {
    gDialog.altTextRadioGroup.selectedItem = gDialog.altTextRadio;
  } else if (hasAltText) {
    gDialog.altTextRadioGroup.selectedItem = gDialog.noAltTextRadio;
  }
  SetAltTextDisabled(
    gDialog.altTextRadioGroup.selectedItem == gDialog.noAltTextRadio
  );

  // setup the height and width widgets
  var width = InitPixelOrPercentMenulist(
    globalElement,
    gInsertNewImage ? null : imageElement,
    "width",
    "widthUnitsMenulist",
    gPixel
  );
  var height = InitPixelOrPercentMenulist(
    globalElement,
    gInsertNewImage ? null : imageElement,
    "height",
    "heightUnitsMenulist",
    gPixel
  );

  // Set actual radio button if both set values are the same as actual
  SetSizeWidgets(width, height);

  gDialog.widthInput.value = gConstrainWidth = width || gActualWidth || "";
  gDialog.heightInput.value = gConstrainHeight = height || gActualHeight || "";

  // set spacing editfields
  gDialog.imagelrInput.value = globalElement.getAttribute("hspace");
  gDialog.imagetbInput.value = globalElement.getAttribute("vspace");

  // dialog.border.value       = globalElement.getAttribute("border");
  var bv = GetHTMLOrCSSStyleValue(globalElement, "border", "border-top-width");
  if (bv.includes("px")) {
    // Strip out the px
    bv = bv.substr(0, bv.indexOf("px"));
  } else if (bv == "thin") {
    bv = "1";
  } else if (bv == "medium") {
    bv = "3";
  } else if (bv == "thick") {
    bv = "5";
  }
  gDialog.border.value = bv;

  // Get alignment setting
  var align = globalElement.getAttribute("align");
  if (align) {
    align = align.toLowerCase();
  }

  switch (align) {
    case "top":
    case "middle":
    case "right":
    case "left":
      gDialog.alignTypeSelect.value = align;
      break;
    default:
      // Default or "bottom"
      gDialog.alignTypeSelect.value = "bottom";
  }

  // Get image map for image
  gImageMap = GetImageMap();

  doOverallEnabling();
  doDimensionEnabling();
}

function SetSizeWidgets(width, height) {
  if (
    !(width || height) ||
    (gActualWidth &&
      gActualHeight &&
      width == gActualWidth &&
      height == gActualHeight)
  ) {
    gDialog.actualSizeRadio.radioGroup.selectedItem = gDialog.actualSizeRadio;
  }

  if (!gDialog.actualSizeRadio.selected) {
    // Decide if user's sizes are in the same ratio as actual sizes
    if (gActualWidth && gActualHeight) {
      if (gActualWidth > gActualHeight) {
        gDialog.constrainCheckbox.checked =
          Math.round((gActualHeight * width) / gActualWidth) == height;
      } else {
        gDialog.constrainCheckbox.checked =
          Math.round((gActualWidth * height) / gActualHeight) == width;
      }
    }
  }
}

// Disable alt text input when "Don't use alt" radio is checked
function SetAltTextDisabled(disable) {
  gDialog.altTextInput.disabled = disable;
}

function GetImageMap() {
  var usemap = globalElement.getAttribute("usemap");
  if (usemap) {
    gCanRemoveImageMap = true;
    let mapname = usemap.substr(1);
    try {
      return GetCurrentEditor().document.querySelector(
        '[name="' + mapname + '"]'
      );
    } catch (e) {}
  } else {
    gCanRemoveImageMap = false;
  }

  return null;
}

function chooseFile() {
  if (gTimerID) {
    clearTimeout(gTimerID);
  }

  // Put focus into the input field
  SetTextboxFocus(gDialog.srcInput);

  GetLocalFileURL("img").then(fileURL => {
    // Always try to relativize local file URLs
    if (gHaveDocumentUrl) {
      fileURL = MakeRelativeUrl(fileURL);
    }

    gDialog.srcInput.value = fileURL;

    SetRelativeCheckbox();
    doOverallEnabling();
    LoadPreviewImage();
  });
}

function PreviewImageLoaded() {
  if (gDialog.PreviewImage) {
    // Image loading has completed -- we can get actual width
    gActualWidth = gDialog.PreviewImage.naturalWidth;
    gActualHeight = gDialog.PreviewImage.naturalHeight;

    if (gActualWidth && gActualHeight) {
      // Use actual size or scale to fit preview if either dimension is too large
      var width = gActualWidth;
      var height = gActualHeight;
      if (gActualWidth > gPreviewImageWidth) {
        width = gPreviewImageWidth;
        height = gActualHeight * (gPreviewImageWidth / gActualWidth);
      }
      if (height > gPreviewImageHeight) {
        height = gPreviewImageHeight;
        width = gActualWidth * (gPreviewImageHeight / gActualHeight);
      }
      gDialog.PreviewImage.width = width;
      gDialog.PreviewImage.height = height;

      gDialog.PreviewWidth.setAttribute("value", gActualWidth);
      gDialog.PreviewHeight.setAttribute("value", gActualHeight);

      document.getElementById("imagePreview").hidden = false;

      SetSizeWidgets(gDialog.widthInput.value, gDialog.heightInput.value);
    }

    if (gDialog.actualSizeRadio.selected) {
      SetActualSize();
    }

    window.sizeToContent();
  }
}

function LoadPreviewImage() {
  var imageSrc = gDialog.srcInput.value.trim();
  if (!imageSrc) {
    return;
  }
  if (isImageDataShortened(imageSrc)) {
    imageSrc = restoredImageData(gDialog.srcInput);
  }

  try {
    // Remove the image URL from image cache so it loads fresh
    //  (if we don't do this, loads after the first will always use image cache
    //   and we won't see image edit changes or be able to get actual width and height)

    // We must have an absolute URL to preview it or remove it from the cache
    imageSrc = MakeAbsoluteUrl(imageSrc);

    if (GetScheme(imageSrc)) {
      let uri = Services.io.newURI(imageSrc);
      if (uri) {
        let imgCache = Cc["@mozilla.org/image/cache;1"].getService(
          Ci.imgICache
        );

        // This returns error if image wasn't in the cache; ignore that
        imgCache.removeEntry(uri);
      }
    }
  } catch (e) {}

  gDialog.PreviewImage.addEventListener("load", PreviewImageLoaded, true);
  gDialog.PreviewImage.src = imageSrc;
}

function SetActualSize() {
  gDialog.widthInput.value = gActualWidth ? gActualWidth : "";
  gDialog.widthUnitsMenulist.selectedIndex = 0;
  gDialog.heightInput.value = gActualHeight ? gActualHeight : "";
  gDialog.heightUnitsMenulist.selectedIndex = 0;
  doDimensionEnabling();
}

function ChangeImageSrc() {
  if (gTimerID) {
    clearTimeout(gTimerID);
  }

  gTimerID = setTimeout(LoadPreviewImage, 800);

  SetRelativeCheckbox();
  doOverallEnabling();
}

function doDimensionEnabling() {
  // Enabled unless "Actual Size" is selected
  var enable = !gDialog.actualSizeRadio.selected;

  // BUG 74145: After input field is disabled,
  //   setting it enabled causes blinking caret to appear
  //   even though focus isn't set to it.
  SetElementEnabledById("heightInput", enable);
  SetElementEnabledById("heightLabel", enable);
  SetElementEnabledById("heightUnitsMenulist", enable);

  SetElementEnabledById("widthInput", enable);
  SetElementEnabledById("widthLabel", enable);
  SetElementEnabledById("widthUnitsMenulist", enable);

  var constrainEnable =
    enable &&
    gDialog.widthUnitsMenulist.selectedIndex == 0 &&
    gDialog.heightUnitsMenulist.selectedIndex == 0;

  SetElementEnabledById("constrainCheckbox", constrainEnable);
}

function doOverallEnabling() {
  var enabled = TrimString(gDialog.srcInput.value) != "";

  SetElementEnabled(gDialog.OkButton, enabled);
  SetElementEnabledById("AdvancedEditButton1", enabled);
  SetElementEnabledById("imagemapLabel", enabled);
  SetElementEnabledById("removeImageMap", gCanRemoveImageMap);
}

function ToggleConstrain() {
  // If just turned on, save the current width and height as basis for constrain ratio
  // Thus clicking on/off lets user say "Use these values as aspect ration"
  if (
    gDialog.constrainCheckbox.checked &&
    !gDialog.constrainCheckbox.disabled &&
    gDialog.widthUnitsMenulist.selectedIndex == 0 &&
    gDialog.heightUnitsMenulist.selectedIndex == 0
  ) {
    gConstrainWidth = Number(TrimString(gDialog.widthInput.value));
    gConstrainHeight = Number(TrimString(gDialog.heightInput.value));
  }
}

function constrainProportions(srcID, destID) {
  var srcElement = document.getElementById(srcID);
  if (!srcElement) {
    return;
  }

  var destElement = document.getElementById(destID);
  if (!destElement) {
    return;
  }

  // always force an integer (whether we are constraining or not)
  forceInteger(srcID);

  if (
    !gActualWidth ||
    !gActualHeight ||
    !(gDialog.constrainCheckbox.checked && !gDialog.constrainCheckbox.disabled)
  ) {
    return;
  }

  // double-check that neither width nor height is in percent mode; bail if so!
  if (
    gDialog.widthUnitsMenulist.selectedIndex != 0 ||
    gDialog.heightUnitsMenulist.selectedIndex != 0
  ) {
    return;
  }

  // This always uses the actual width and height ratios
  // which is kind of funky if you change one number without the constrain
  // and then turn constrain on and change a number
  // I prefer the old strategy (below) but I can see some merit to this solution
  if (srcID == "widthInput") {
    destElement.value = Math.round(
      (srcElement.value * gActualHeight) / gActualWidth
    );
  } else {
    destElement.value = Math.round(
      (srcElement.value * gActualWidth) / gActualHeight
    );
  }

  /*
  // With this strategy, the width and height ratio
  //   can be reset to whatever the user entered.
  if (srcID == "widthInput") {
    destElement.value = Math.round( srcElement.value * gConstrainHeight / gConstrainWidth );
  } else {
    destElement.value = Math.round( srcElement.value * gConstrainWidth / gConstrainHeight );
  }
  */
}

function removeImageMap() {
  gRemoveImageMap = true;
  gCanRemoveImageMap = false;
  SetElementEnabledById("removeImageMap", false);
}

function SwitchToValidatePanel() {
  if (
    gDialog.tabBox &&
    gValidateTab &&
    gDialog.tabBox.selectedTab != gValidateTab
  ) {
    gDialog.tabBox.selectedTab = gValidateTab;
  }
}

// Get data from widgets, validate, and set for the global element
//   accessible to AdvancedEdit() [in EdDialogCommon.js]
function ValidateImage() {
  var editor = GetCurrentEditor();
  if (!editor) {
    return false;
  }

  gValidateTab = gDialog.tabLocation;
  if (!gDialog.srcInput.value) {
    Services.prompt.alert(
      window,
      GetString("Alert"),
      GetString("MissingImageError")
    );
    SwitchToValidatePanel();
    gDialog.srcInput.focus();
    return false;
  }

  // We must convert to "file:///" or "http://" format else image doesn't load!
  let src = gDialog.srcInput.value.trim();

  if (isImageDataShortened(src)) {
    src = restoredImageData(gDialog.srcInput);
  } else {
    var checkbox = document.getElementById("MakeRelativeCheckbox");
    try {
      if (checkbox && !checkbox.checked) {
        src = Services.uriFixup.createFixupURI(
          src,
          Ci.nsIURIFixup.FIXUP_FLAG_NONE
        ).spec;
      }
    } catch (e) {}

    globalElement.setAttribute("src", src);
  }

  let title = gDialog.titleInput.value.trim();
  if (title) {
    globalElement.setAttribute("title", title);
  } else {
    globalElement.removeAttribute("title");
  }

  // Force user to enter Alt text only if "Alternate text" radio is checked
  // Don't allow just spaces in alt text
  var alt = "";
  var useAlt = gDialog.altTextRadioGroup.selectedItem == gDialog.altTextRadio;
  if (useAlt) {
    alt = TrimString(gDialog.altTextInput.value);
  }

  if (alt || !useAlt) {
    globalElement.setAttribute("alt", alt);
  } else if (!gDoAltTextError) {
    globalElement.removeAttribute("alt");
  } else {
    Services.prompt.alert(window, GetString("Alert"), GetString("NoAltText"));
    SwitchToValidatePanel();
    gDialog.altTextInput.focus();
    return false;
  }

  var width = "";
  var height = "";

  gValidateTab = gDialog.tabDimensions;
  if (!gDialog.actualSizeRadio.selected) {
    // Get user values for width and height
    width = ValidateNumber(
      gDialog.widthInput,
      gDialog.widthUnitsMenulist,
      1,
      gMaxPixels,
      globalElement,
      "width",
      false,
      true
    );
    if (gValidationError) {
      return false;
    }

    height = ValidateNumber(
      gDialog.heightInput,
      gDialog.heightUnitsMenulist,
      1,
      gMaxPixels,
      globalElement,
      "height",
      false,
      true
    );
    if (gValidationError) {
      return false;
    }
  }

  // We always set the width and height attributes, even if same as actual.
  //  This speeds up layout of pages since sizes are known before image is loaded
  if (!width) {
    width = gActualWidth;
  }
  if (!height) {
    height = gActualHeight;
  }

  // Remove existing width and height only if source changed
  //  and we couldn't obtain actual dimensions
  var srcChanged = src != gOriginalSrc;
  if (width) {
    editor.setAttributeOrEquivalent(globalElement, "width", width, true);
  } else if (srcChanged) {
    editor.removeAttributeOrEquivalent(globalElement, "width", true);
  }

  if (height) {
    editor.setAttributeOrEquivalent(globalElement, "height", height, true);
  } else if (srcChanged) {
    editor.removeAttributeOrEquivalent(globalElement, "height", true);
  }

  // spacing attributes
  gValidateTab = gDialog.tabBorder;
  ValidateNumber(
    gDialog.imagelrInput,
    null,
    0,
    gMaxPixels,
    globalElement,
    "hspace",
    false,
    true,
    true
  );
  if (gValidationError) {
    return false;
  }

  ValidateNumber(
    gDialog.imagetbInput,
    null,
    0,
    gMaxPixels,
    globalElement,
    "vspace",
    false,
    true
  );
  if (gValidationError) {
    return false;
  }

  // note this is deprecated and should be converted to stylesheets
  ValidateNumber(
    gDialog.border,
    null,
    0,
    gMaxPixels,
    globalElement,
    "border",
    false,
    true
  );
  if (gValidationError) {
    return false;
  }

  // Default or setting "bottom" means don't set the attribute
  // Note that the attributes "left" and "right" are opposite
  //  of what we use in the UI, which describes where the TEXT wraps,
  //  not the image location (which is what the HTML describes)
  switch (gDialog.alignTypeSelect.value) {
    case "top":
    case "middle":
    case "right":
    case "left":
      editor.setAttributeOrEquivalent(
        globalElement,
        "align",
        gDialog.alignTypeSelect.value,
        true
      );
      break;
    default:
      try {
        editor.removeAttributeOrEquivalent(globalElement, "align", true);
      } catch (e) {}
  }

  return true;
}