Browse Source

merge from examcloud-T:feature-20200316-fs

deason 5 years ago
parent
commit
224ccbb7e2
100 changed files with 4950 additions and 432 deletions
  1. 9 0
      jenkins-dev.sh
  2. 1 0
      public/ckeditor/plugins/base64image/dialogs/base64image.js
  3. 6 6
      public/ckeditor/plugins/base64image/plugin.js
  4. BIN
      public/ckeditor/plugins/image2/dev/assets/image1.jpg
  5. BIN
      public/ckeditor/plugins/image2/dev/assets/image2.jpg
  6. 35 0
      public/ckeditor/plugins/image2/dev/contents.css
  7. 339 0
      public/ckeditor/plugins/image2/dev/image2.html
  8. 553 0
      public/ckeditor/plugins/image2/dialogs/image2.js
  9. BIN
      public/ckeditor/plugins/image2/icons/hidpi/image.png
  10. BIN
      public/ckeditor/plugins/image2/icons/image.png
  11. 21 0
      public/ckeditor/plugins/image2/lang/af.js
  12. 21 0
      public/ckeditor/plugins/image2/lang/ar.js
  13. 21 0
      public/ckeditor/plugins/image2/lang/az.js
  14. 21 0
      public/ckeditor/plugins/image2/lang/bg.js
  15. 21 0
      public/ckeditor/plugins/image2/lang/bn.js
  16. 21 0
      public/ckeditor/plugins/image2/lang/bs.js
  17. 21 0
      public/ckeditor/plugins/image2/lang/ca.js
  18. 21 0
      public/ckeditor/plugins/image2/lang/cs.js
  19. 21 0
      public/ckeditor/plugins/image2/lang/cy.js
  20. 21 0
      public/ckeditor/plugins/image2/lang/da.js
  21. 21 0
      public/ckeditor/plugins/image2/lang/de-ch.js
  22. 21 0
      public/ckeditor/plugins/image2/lang/de.js
  23. 21 0
      public/ckeditor/plugins/image2/lang/el.js
  24. 21 0
      public/ckeditor/plugins/image2/lang/en-au.js
  25. 21 0
      public/ckeditor/plugins/image2/lang/en-ca.js
  26. 21 0
      public/ckeditor/plugins/image2/lang/en-gb.js
  27. 21 0
      public/ckeditor/plugins/image2/lang/en.js
  28. 21 0
      public/ckeditor/plugins/image2/lang/eo.js
  29. 21 0
      public/ckeditor/plugins/image2/lang/es-mx.js
  30. 21 0
      public/ckeditor/plugins/image2/lang/es.js
  31. 21 0
      public/ckeditor/plugins/image2/lang/et.js
  32. 21 0
      public/ckeditor/plugins/image2/lang/eu.js
  33. 21 0
      public/ckeditor/plugins/image2/lang/fa.js
  34. 21 0
      public/ckeditor/plugins/image2/lang/fi.js
  35. 21 0
      public/ckeditor/plugins/image2/lang/fo.js
  36. 21 0
      public/ckeditor/plugins/image2/lang/fr-ca.js
  37. 21 0
      public/ckeditor/plugins/image2/lang/fr.js
  38. 21 0
      public/ckeditor/plugins/image2/lang/gl.js
  39. 21 0
      public/ckeditor/plugins/image2/lang/gu.js
  40. 21 0
      public/ckeditor/plugins/image2/lang/he.js
  41. 21 0
      public/ckeditor/plugins/image2/lang/hi.js
  42. 21 0
      public/ckeditor/plugins/image2/lang/hr.js
  43. 21 0
      public/ckeditor/plugins/image2/lang/hu.js
  44. 21 0
      public/ckeditor/plugins/image2/lang/id.js
  45. 21 0
      public/ckeditor/plugins/image2/lang/is.js
  46. 21 0
      public/ckeditor/plugins/image2/lang/it.js
  47. 21 0
      public/ckeditor/plugins/image2/lang/ja.js
  48. 21 0
      public/ckeditor/plugins/image2/lang/ka.js
  49. 21 0
      public/ckeditor/plugins/image2/lang/km.js
  50. 21 0
      public/ckeditor/plugins/image2/lang/ko.js
  51. 21 0
      public/ckeditor/plugins/image2/lang/ku.js
  52. 21 0
      public/ckeditor/plugins/image2/lang/lt.js
  53. 21 0
      public/ckeditor/plugins/image2/lang/lv.js
  54. 21 0
      public/ckeditor/plugins/image2/lang/mk.js
  55. 21 0
      public/ckeditor/plugins/image2/lang/mn.js
  56. 21 0
      public/ckeditor/plugins/image2/lang/ms.js
  57. 21 0
      public/ckeditor/plugins/image2/lang/nb.js
  58. 21 0
      public/ckeditor/plugins/image2/lang/nl.js
  59. 21 0
      public/ckeditor/plugins/image2/lang/no.js
  60. 21 0
      public/ckeditor/plugins/image2/lang/oc.js
  61. 21 0
      public/ckeditor/plugins/image2/lang/pl.js
  62. 21 0
      public/ckeditor/plugins/image2/lang/pt-br.js
  63. 21 0
      public/ckeditor/plugins/image2/lang/pt.js
  64. 21 0
      public/ckeditor/plugins/image2/lang/ro.js
  65. 21 0
      public/ckeditor/plugins/image2/lang/ru.js
  66. 21 0
      public/ckeditor/plugins/image2/lang/si.js
  67. 21 0
      public/ckeditor/plugins/image2/lang/sk.js
  68. 21 0
      public/ckeditor/plugins/image2/lang/sl.js
  69. 21 0
      public/ckeditor/plugins/image2/lang/sq.js
  70. 21 0
      public/ckeditor/plugins/image2/lang/sr-latn.js
  71. 21 0
      public/ckeditor/plugins/image2/lang/sr.js
  72. 21 0
      public/ckeditor/plugins/image2/lang/sv.js
  73. 21 0
      public/ckeditor/plugins/image2/lang/th.js
  74. 21 0
      public/ckeditor/plugins/image2/lang/tr.js
  75. 21 0
      public/ckeditor/plugins/image2/lang/tt.js
  76. 21 0
      public/ckeditor/plugins/image2/lang/ug.js
  77. 21 0
      public/ckeditor/plugins/image2/lang/uk.js
  78. 21 0
      public/ckeditor/plugins/image2/lang/vi.js
  79. 21 0
      public/ckeditor/plugins/image2/lang/zh-cn.js
  80. 21 0
      public/ckeditor/plugins/image2/lang/zh.js
  81. 1783 0
      public/ckeditor/plugins/image2/plugin.js
  82. BIN
      public/ckeditor/plugins/image2/samples/assets/image1.jpg
  83. BIN
      public/ckeditor/plugins/image2/samples/assets/image2.jpg
  84. 44 0
      public/ckeditor/plugins/image2/samples/image2.html
  85. 1 0
      public/ckeditor/plugins/pastebase64/plugin.js
  86. 6 3
      src/components/ckeditor.vue
  87. 2 0
      src/modules/basic/view/course.vue
  88. 13 9
      src/modules/basic/view/school_config.vue
  89. 2 2
      src/modules/examwork/view/examInfo.vue
  90. 68 248
      src/modules/examwork/view/examStudent.vue
  91. 44 5
      src/modules/examwork/view/offlineExam.vue
  92. 1 19
      src/modules/examwork/view/onlineExam.vue
  93. 14 24
      src/modules/marking/views/TpMain.vue
  94. 0 5
      src/modules/oe/views/absent.vue
  95. 25 107
      src/modules/oe/views/examScheduling.vue
  96. 6 3
      src/modules/questions/component/ckeditor.vue
  97. 5 0
      src/modules/questions/routes/routes.js
  98. 1 1
      src/modules/questions/views/EditPaper.vue
  99. 1 0
      src/modules/questions/views/ExportStructure.vue
  100. 521 0
      src/modules/questions/views/ExportTemplate.vue

+ 9 - 0
jenkins-dev.sh

@@ -0,0 +1,9 @@
+#!/bin/bash
+pwd
+
+yarn
+npm run build
+
+rm -rf ~/project/examcloud/static/examcloud-admin-web/dist/*
+mkdir -p ~/project/examcloud/static/examcloud-admin-web
+cp -r  dist ~/project/examcloud/static/examcloud-admin-web/

+ 1 - 0
public/ckeditor/plugins/base64image/dialogs/base64image.js

@@ -521,6 +521,7 @@ CKEDITOR.dialog.add("base64imageDialog", function(editor) {
       /* Resize image */
       if (editor.plugins.imageresize)
         editor.plugins.imageresize.resize(editor, newImg, 800, 800);
+      editor.setData(editor.getData());
     },
 
     /* Dialog form */

+ 6 - 6
public/ckeditor/plugins/base64image/plugin.js

@@ -1,6 +1,6 @@
 /*
  * Base64Image Plugin for CKEditor (http://github.com/nmmf/base64image)
- * Created by ALL-INKL.COM - Neue Medien M�nnich - 04. Feb 2014
+ * Created by ALL-INKL.COM - Neue Medien M锟絥nich - 04. Feb 2014
  * Licensed under the terms of GPL, LGPL and MPL licenses.
  */
 CKEDITOR.plugins.add("base64image", {
@@ -92,11 +92,11 @@ CKEDITOR.plugins.add("base64image", {
       pluginName,
       new CKEDITOR.dialogCommand(pluginName, {
         allowedContent: allowed,
-        requiredContent: required,
-        contentTransformations: [
-          ["img{width}: sizeToStyle", "img[width]: sizeToAttribute"],
-          ["img{float}: alignmentToStyle", "img[align]: alignmentToAttribute"]
-        ]
+        requiredContent: required
+        // contentTransformations: [
+        //   ["img{width}: sizeToStyle", "img[width]: sizeToAttribute"],
+        //   ["img{float}: alignmentToStyle", "img[align]: alignmentToAttribute"]
+        // ]
       })
     );
     editor.on("doubleclick", function(evt) {

BIN
public/ckeditor/plugins/image2/dev/assets/image1.jpg


BIN
public/ckeditor/plugins/image2/dev/assets/image2.jpg


+ 35 - 0
public/ckeditor/plugins/image2/dev/contents.css

@@ -0,0 +1,35 @@
+/*
+Copyright (c) 2003-2019, CKSource - Frederico Knabben. All rights reserved.
+For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license
+*/
+
+.cke_widget_wrapper:hover:after {
+	content: "id: " attr(data-cke-widget-id);
+	position: absolute;
+	top: 0;
+	right: 0;
+	padding: 2px 4px;
+	background: #EEE;
+	border: solid 1px #DDD;
+	border-radius: 2px;
+	color: #BBB;
+	font: bold 10px sans-serif;
+}
+
+.align-left {
+	float: left;
+	margin-right: 20px;
+}
+
+.align-right {
+	float: right;
+	margin-left: 20px;
+}
+
+.align-center {
+	text-align: center;
+}
+
+.align-center > figure {
+	display: inline-block;
+}

+ 339 - 0
public/ckeditor/plugins/image2/dev/image2.html

@@ -0,0 +1,339 @@
+<!DOCTYPE html>
+<!--
+Copyright (c) 2003-2019, CKSource - Frederico Knabben. All rights reserved.
+For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license
+-->
+<html>
+<head>
+	<meta charset="utf-8">
+	<title>Widget Image &mdash; CKEditor Sample</title>
+	<script src="../../../ckeditor.js"></script>
+	<script src="../../../dev/console/console.js"></script>
+	<script src="../../../dev/console/focusconsole.js"></script>
+	<script src="../../widget/dev/console.js"></script>
+	<script>
+		if ( CKEDITOR.env.ie && CKEDITOR.env.version < 9 )
+			CKEDITOR.tools.enableHtml5Elements( document );
+
+		var editor;
+
+		// The instanceReady event is fired, when an instance of CKEditor has finished
+		// its initialization.
+		CKEDITOR.on( 'instanceReady', function( ev ) {
+			editor = ev.editor;
+
+			// Show this "on" button.
+			document.getElementById( 'readOnlyOn' ).style.display = '';
+
+			// Event fired when the readOnly property changes.
+			editor.on( 'readOnly', function() {
+				document.getElementById( 'readOnlyOn' ).style.display = this.readOnly ? 'none' : '';
+				document.getElementById( 'readOnlyOff' ).style.display = this.readOnly ? '' : 'none';
+			});
+		});
+
+		function toggleReadOnly( isReadOnly ) {
+			// Change the read-only state of the editor.
+			// https://ckeditor.com/docs/ckeditor4/latest/api/CKEDITOR_editor.html#method-setReadOnly
+			editor.setReadOnly( isReadOnly );
+		}
+
+	</script>
+	<link href="../../../samples/old/sample.css" rel="stylesheet">
+
+	<style>
+
+		body {
+			font-size: 13px;
+		}
+		.editable {
+			padding: 20px;
+			border: 2px solid #dfdfdf;
+			overflow: auto;
+		}
+
+		body p {
+			line-height: 1.8em;
+		}
+
+		/* Reset some styles from sample.css */
+		.cke_editable.cke_editable_inline
+		{
+			cursor: auto;
+		}
+		.cke_editable.cke_editable_inline.cke_focus
+		{
+			box-shadow: none;
+			background: inherit;
+			cursor: auto;
+		}
+
+	</style>
+	<link href="contents.css" rel="stylesheet">
+	<link href="../../../contents.css" rel="stylesheet">
+</head>
+<body>
+	<h1 class="samples">
+		<a href="../../../samples/old/index.html">CKEditor Samples</a> &raquo; Widget Image
+	</h1>
+
+	<h2>Classic (iframe-based) Sample</h2>
+
+	<textarea id="editor1" cols="10" rows="10">
+		<h1>Apollo 11</h1>
+
+		<figure class="image" style="float: right">
+			<img alt="Saturn V" src="assets/image1.jpg" width="200" data-foo="*********" data-bar="@@@@@@@@" />
+			<figcaption>Roll out of Saturn V on launch pad</figcaption>
+		</figure>
+
+		<p><strong>Apollo 11</strong> was the spaceflight that landed the first humans, Americans <a href="http://en.wikipedia.org/wiki/Neil_Armstrong" title="Neil Armstrong">Neil Armstrong</a> and <a href="http://en.wikipedia.org/wiki/Buzz_Aldrin" title="Buzz Aldrin">Buzz Aldrin</a>, on the Moon on July 20, 1969, at 20:18 UTC. Armstrong became the first to step onto the lunar surface 6 hours later on July 21 at 02:56 UTC.</p>
+
+		<p>Armstrong spent about <s>three and a half</s> two and a half hours outside the spacecraft, Aldrin slightly less; and together they collected 47.5 pounds (21.5&nbsp;kg) of lunar material for return to Earth. A third member of the mission, <a href="http://en.wikipedia.org/wiki/Michael_Collins_(astronaut)" title="Michael Collins (astronaut)">Michael Collins</a>, piloted the <a href="http://en.wikipedia.org/wiki/Apollo_Command/Service_Module" title="Apollo Command/Service Module">command</a> spacecraft alone in lunar orbit until Armstrong and Aldrin returned to it for the trip back to Earth.</p>
+
+		<h2>Broadcasting and <em>quotes</em> <a id="quotes" name="quotes"></a></h2>
+
+		<p>Broadcast on live TV to a world-wide audience, Armstrong stepped onto the lunar surface and described the event as:</p>
+
+		<blockquote>
+		<p>One small step for [a] man, one giant leap for mankind.</p>
+		</blockquote>
+
+		<p>Apollo 11 effectively ended the <a href="http://en.wikipedia.org/wiki/Space_Race" title="Space Race">Space Race</a> and fulfilled a national goal proposed in 1961 by the late U.S. President <a href="http://en.wikipedia.org/wiki/John_F._Kennedy" title="John F. Kennedy">John F. Kennedy</a> in a speech before the United States Congress:</p>
+
+		<blockquote>
+		<p>[...] before this decade is out, of landing a man on the Moon and returning him safely to the Earth.</p>
+		</blockquote>
+
+		<figure class="image" style="float: right">
+			<img alt="The Eagle" src="assets/image2.jpg" style="width: 200px" />
+			<figcaption>The Eagle in lunar orbit</figcaption>
+		</figure>
+
+		<h2>Technical details <a id="tech-details" name="tech-details"></a></h2>
+
+		<p>Launched by a <strong>Saturn V</strong> rocket from <a href="http://en.wikipedia.org/wiki/Kennedy_Space_Center" title="Kennedy Space Center">Kennedy Space Center</a> in Merritt Island, Florida on July 16, Apollo 11 was the fifth manned mission of <a href="http://en.wikipedia.org/wiki/NASA" title="NASA">NASA</a>&#39;s Apollo program. The Apollo spacecraft had three parts:</p>
+
+		<ol>
+			<li><strong>Command Module</strong> with a cabin for the three astronauts which was the only part which landed back on Earth</li>
+			<li><strong>Service Module</strong> which supported the Command Module with propulsion, electrical power, oxygen and water</li>
+			<li><strong>Lunar Module</strong> for landing on the Moon.</li>
+		</ol>
+
+		<p>After being sent to the Moon by the Saturn V&#39;s upper stage, the astronauts separated the spacecraft from it and travelled for three days until they entered into lunar orbit. Armstrong and Aldrin then moved into the Lunar Module and landed in the <a href="http://en.wikipedia.org/wiki/Mare_Tranquillitatis" title="Mare Tranquillitatis">Sea of Tranquility</a>. They stayed a total of about 21 and a half hours on the lunar surface. After lifting off in the upper part of the Lunar Module and rejoining Collins in the Command Module, they returned to Earth and landed in the <a href="http://en.wikipedia.org/wiki/Pacific_Ocean" title="Pacific Ocean">Pacific Ocean</a> on July 24.</p>
+
+		<p style="text-align:center">
+			<img alt="Saturn V" src="assets/image1.jpg" width="200" />
+		</p>
+
+		<hr />
+		<p style="text-align:right"><small>Source: <a href="http://en.wikipedia.org/wiki/Apollo_11">Wikipedia.org</a></small></p>
+	</textarea>
+
+	<h2>Inline Sample</h2>
+
+	<div id="editor2" contenteditable="true" class="editable">
+		<h2>Apollo 11</h2>
+
+		<figure class="image" style="float: right">
+			<img alt="Saturn V" src="assets/image1.jpg" width="200" />
+			<figcaption>Roll out of Saturn V on launch pad</figcaption>
+		</figure>
+
+		<p><strong>Apollo 11</strong> was the spaceflight that landed the first humans, Americans <a href="http://en.wikipedia.org/wiki/Neil_Armstrong" title="Neil Armstrong">Neil Armstrong</a> and <a href="http://en.wikipedia.org/wiki/Buzz_Aldrin" title="Buzz Aldrin">Buzz Aldrin</a>, on the Moon on July 20, 1969, at 20:18 UTC. Armstrong became the first to step onto the lunar surface 6 hours later on July 21 at 02:56 UTC.</p>
+
+		<p>Armstrong spent about <s>three and a half</s> two and a half hours outside the spacecraft, Aldrin slightly less; and together they collected 47.5 pounds (21.5&nbsp;kg) of lunar material for return to Earth. A third member of the mission, <a href="http://en.wikipedia.org/wiki/Michael_Collins_(astronaut)" title="Michael Collins (astronaut)">Michael Collins</a>, piloted the <a href="http://en.wikipedia.org/wiki/Apollo_Command/Service_Module" title="Apollo Command/Service Module">command</a> spacecraft alone in lunar orbit until Armstrong and Aldrin returned to it for the trip back to Earth.</p>
+
+		<h2>Broadcasting and <em>quotes</em> <a id="quotes" name="quotes"></a></h2>
+
+		<p>Broadcast on live TV to a world-wide audience, Armstrong stepped onto the lunar surface and described the event as:</p>
+
+		<blockquote>
+		<p>One small step for [a] man, one giant leap for mankind.</p>
+		</blockquote>
+
+		<p>Apollo 11 effectively ended the <a href="http://en.wikipedia.org/wiki/Space_Race" title="Space Race">Space Race</a> and fulfilled a national goal proposed in 1961 by the late U.S. President <a href="http://en.wikipedia.org/wiki/John_F._Kennedy" title="John F. Kennedy">John F. Kennedy</a> in a speech before the United States Congress:</p>
+
+		<blockquote>
+		<p>[...] before this decade is out, of landing a man on the Moon and returning him safely to the Earth.</p>
+		</blockquote>
+
+		<figure class="image" style="float: right">
+			<img alt="The Eagle" src="assets/image2.jpg" style="width: 200px" />
+			<figcaption>The Eagle in lunar orbit</figcaption>
+		</figure>
+
+		<h2>Technical details <a id="tech-details" name="tech-details"></a></h2>
+
+		<p>Launched by a <strong>Saturn V</strong> rocket from <a href="http://en.wikipedia.org/wiki/Kennedy_Space_Center" title="Kennedy Space Center">Kennedy Space Center</a> in Merritt Island, Florida on July 16, Apollo 11 was the fifth manned mission of <a href="http://en.wikipedia.org/wiki/NASA" title="NASA">NASA</a>&#39;s Apollo program. The Apollo spacecraft had three parts:</p>
+
+		<ol>
+			<li><strong>Command Module</strong> with a cabin for the three astronauts which was the only part which landed back on Earth</li>
+			<li><strong>Service Module</strong> which supported the Command Module with propulsion, electrical power, oxygen and water</li>
+			<li><strong>Lunar Module</strong> for landing on the Moon.</li>
+		</ol>
+
+		<p>After being sent to the Moon by the Saturn V&#39;s upper stage, the astronauts separated the spacecraft from it and travelled for three days until they entered into lunar orbit. Armstrong and Aldrin then moved into the Lunar Module and landed in the <a href="http://en.wikipedia.org/wiki/Mare_Tranquillitatis" title="Mare Tranquillitatis">Sea of Tranquility</a>. They stayed a total of about 21 and a half hours on the lunar surface. After lifting off in the upper part of the Lunar Module and rejoining Collins in the Command Module, they returned to Earth and landed in the <a href="http://en.wikipedia.org/wiki/Pacific_Ocean" title="Pacific Ocean">Pacific Ocean</a> on July 24.</p>
+
+		<p style="text-align:center">
+			<img alt="Saturn V" src="assets/image1.jpg" width="200" />
+		</p>
+
+		<hr />
+		<p style="text-align:right"><small>Source: <a href="http://en.wikipedia.org/wiki/Apollo_11">Wikipedia.org</a></small></p>
+	</div>
+
+	<h2>Div Editing Area Sample</h2>
+
+	<textarea id="editor3" cols="10" rows="10">
+		<h1>Apollo 11</h1>
+
+		<figure class="caption" style="float: right">
+			<img alt="Saturn V" src="assets/image1.jpg" width="200" />
+			<figcaption>Roll out of Saturn V on launch pad</figcaption>
+		</figure>
+
+		<p><strong>Apollo 11</strong> was the spaceflight that landed the first humans, Americans <a href="http://en.wikipedia.org/wiki/Neil_Armstrong" title="Neil Armstrong">Neil Armstrong</a> and <a href="http://en.wikipedia.org/wiki/Buzz_Aldrin" title="Buzz Aldrin">Buzz Aldrin</a>, on the Moon on July 20, 1969, at 20:18 UTC. Armstrong became the first to step onto the lunar surface 6 hours later on July 21 at 02:56 UTC.</p>
+
+		<p>Armstrong spent about <s>three and a half</s> two and a half hours outside the spacecraft, Aldrin slightly less; and together they collected 47.5 pounds (21.5&nbsp;kg) of lunar material for return to Earth. A third member of the mission, <a href="http://en.wikipedia.org/wiki/Michael_Collins_(astronaut)" title="Michael Collins (astronaut)">Michael Collins</a>, piloted the <a href="http://en.wikipedia.org/wiki/Apollo_Command/Service_Module" title="Apollo Command/Service Module">command</a> spacecraft alone in lunar orbit until Armstrong and Aldrin returned to it for the trip back to Earth.</p>
+
+		<h2>Broadcasting and <em>quotes</em> <a id="quotes" name="quotes"></a></h2>
+
+		<p>Broadcast on live TV to a world-wide audience, Armstrong stepped onto the lunar surface and described the event as:</p>
+
+		<blockquote>
+		<p>One small step for [a] man, one giant leap for mankind.</p>
+		</blockquote>
+
+		<p>Apollo 11 effectively ended the <a href="http://en.wikipedia.org/wiki/Space_Race" title="Space Race">Space Race</a> and fulfilled a national goal proposed in 1961 by the late U.S. President <a href="http://en.wikipedia.org/wiki/John_F._Kennedy" title="John F. Kennedy">John F. Kennedy</a> in a speech before the United States Congress:</p>
+
+		<blockquote>
+		<p>[...] before this decade is out, of landing a man on the Moon and returning him safely to the Earth.</p>
+		</blockquote>
+
+		<figure class="caption" style="float: right">
+			<img alt="The Eagle" src="assets/image2.jpg" style="width: 200px" />
+			<figcaption>The Eagle in lunar orbit</figcaption>
+		</figure>
+
+		<h2>Technical Details <a id="tech-details" name="tech-details"></a></h2>
+
+		<p>Launched by a <strong>Saturn V</strong> rocket from <a href="http://en.wikipedia.org/wiki/Kennedy_Space_Center" title="Kennedy Space Center">Kennedy Space Center</a> in Merritt Island, Florida on July 16, Apollo 11 was the fifth manned mission of <a href="http://en.wikipedia.org/wiki/NASA" title="NASA">NASA</a>&#39;s Apollo program. The Apollo spacecraft had three parts:</p>
+
+		<ol>
+			<li><strong>Command Module</strong> with a cabin for the three astronauts which was the only part which landed back on Earth</li>
+			<li><strong>Service Module</strong> which supported the Command Module with propulsion, electrical power, oxygen and water</li>
+			<li><strong>Lunar Module</strong> for landing on the Moon.</li>
+		</ol>
+
+		<p>After being sent to the Moon by the Saturn V&#39;s upper stage, the astronauts separated the spacecraft from it and travelled for three days until they entered into lunar orbit. Armstrong and Aldrin then moved into the Lunar Module and landed in the <a href="http://en.wikipedia.org/wiki/Mare_Tranquillitatis" title="Mare Tranquillitatis">Sea of Tranquility</a>. They stayed a total of about 21 and a half hours on the lunar surface. After lifting off in the upper part of the Lunar Module and rejoining Collins in the Command Module, they returned to Earth and landed in the <a href="http://en.wikipedia.org/wiki/Pacific_Ocean" title="Pacific Ocean">Pacific Ocean</a> on July 24.</p>
+
+		<p style="text-align:center">
+			<img alt="Saturn V" src="assets/image1.jpg" width="200" />
+		</p>
+
+		<hr />
+		<p style="text-align:right"><small>Source: <a href="http://en.wikipedia.org/wiki/Apollo_11">Wikipedia.org</a></small></p>
+	</textarea>
+
+	<h2>alignClasses samples</h2>
+
+	<textarea id="editor4" cols="10" rows="10">
+		<h1>Apollo 11</h1>
+
+		<figure class="align-left image">
+			<img alt="Saturn V" src="assets/image1.jpg" width="200" data-foo="*********" data-bar="@@@@@@@@" />
+			<figcaption>Roll out of Saturn V on launch pad</figcaption>
+		</figure>
+
+		<p><strong>Apollo 11</strong> was the spaceflight that landed the first humans, Americans <a href="http://en.wikipedia.org/wiki/Neil_Armstrong" title="Neil Armstrong">Neil Armstrong</a> and <a href="http://en.wikipedia.org/wiki/Buzz_Aldrin" title="Buzz Aldrin">Buzz Aldrin</a>, on the Moon on July 20, 1969, at 20:18 UTC. Armstrong became the first to step onto the lunar surface 6 hours later on July 21 at 02:56 UTC.</p>
+
+		<blockquote>
+		<p>[...] before this decade is out, of landing a man on the Moon and returning him safely to the Earth.</p>
+		</blockquote>
+
+		<figure class="align-right image">
+			<img alt="The Eagle" src="assets/image2.jpg" style="width: 200px" />
+			<figcaption>The Eagle in lunar orbit</figcaption>
+		</figure>
+
+		<h2>Technical details <a id="tech-details" name="tech-details"></a></h2>
+
+		<p>Launched by a <strong>Saturn V</strong> rocket from <a href="http://en.wikipedia.org/wiki/Kennedy_Space_Center" title="Kennedy Space Center">Kennedy Space Center</a> in Merritt Island, Florida on July 16, Apollo 11 was the fifth manned mission of <a href="http://en.wikipedia.org/wiki/NASA" title="NASA">NASA</a>&#39;s Apollo program. The Apollo spacecraft had three parts:</p>
+
+		<ol>
+			<li><strong>Command Module</strong> with a cabin for the three astronauts which was the only part which landed back on Earth</li>
+			<li><strong>Service Module</strong> which supported the Command Module with propulsion, electrical power, oxygen and water</li>
+			<li><strong>Lunar Module</strong> for landing on the Moon.</li>
+		</ol>
+
+		<p>After being sent to the Moon by the Saturn V&#39;s upper stage, the astronauts separated the spacecraft from it and travelled for three days until they entered into lunar orbit. Armstrong and Aldrin then moved into the Lunar Module and landed in the <a href="http://en.wikipedia.org/wiki/Mare_Tranquillitatis" title="Mare Tranquillitatis">Sea of Tranquility</a>. They stayed a total of about 21 and a half hours on the lunar surface. After lifting off in the upper part of the Lunar Module and rejoining Collins in the Command Module, they returned to Earth and landed in the <a href="http://en.wikipedia.org/wiki/Pacific_Ocean" title="Pacific Ocean">Pacific Ocean</a> on July 24.</p>
+
+		<p class="align-center">
+			<img alt="Saturn V" src="assets/image1.jpg" width="200" />
+		</p>
+
+		<hr />
+		<p style="text-align:right"><small>Source: <a href="http://en.wikipedia.org/wiki/Apollo_11">Wikipedia.org</a></small></p>
+	</textarea>
+
+	<p>
+		<input id="readOnlyOn" onclick="toggleReadOnly( true );" type="button" value="Make it read-only" style="display:none">
+		<input id="readOnlyOff" onclick="toggleReadOnly( false );" type="button" value="Make it editable again" style="display:none">
+	</p>
+
+	<script>
+
+		CKEDITOR.disableAutoInline = true;
+
+		CKEDITOR.replace( 'editor1', {
+			extraPlugins: 'image2',
+			height: 600,
+			contentsCss: [ '../../../contents.css', 'contents.css' ],
+			extraAllowedContent: 'img[data-foo,data-bar]',
+
+			filebrowserBrowseUrl: '/ckfinder/ckfinder.html',
+			filebrowserImageBrowseUrl: '/ckfinder/ckfinder.html?Type=Images',
+			filebrowserUploadUrl: '/ckfinder/core/connector/php/connector.php?command=QuickUpload&type=Files',
+			filebrowserImageUploadUrl: '/ckfinder/core/connector/php/connector.php?command=QuickUpload&type=Images',
+		} );
+
+		CKEDITOR.inline( 'editor2', {
+			extraPlugins: 'image2,sourcedialog'
+		} );
+
+		CKEDITOR.replace( 'editor3', {
+			extraPlugins: 'image2,divarea',
+			height: 600
+		} );
+
+		CKEDITOR.replace( 'editor4', {
+			extraPlugins: 'image2',
+			image2_alignClasses: [ 'align-left', 'align-center', 'align-right' ],
+			contentsCss: [ '../../../contents.css', 'contents.css' ],
+			height: 600
+		} );
+
+		CKCONSOLE.create( 'widget', { editor: 'editor1' } );
+		CKCONSOLE.create( 'focus', { editor: 'editor1' } );
+		CKCONSOLE.create( 'widget', { editor: 'editor2', folded: true } );
+		CKCONSOLE.create( 'focus', { editor: 'editor2', folded: true } );
+		CKCONSOLE.create( 'widget', { editor: 'editor3' } );
+		CKCONSOLE.create( 'focus', { editor: 'editor3' } );
+		CKCONSOLE.create( 'widget', { editor: 'editor4' } );
+		CKCONSOLE.create( 'focus', { editor: 'editor4' } );
+
+	</script>
+
+	<div id="footer">
+		<hr>
+		<p>
+			CKEditor - The text editor for the Internet - <a class="samples" href="https://ckeditor.com/">https://ckeditor.com</a>
+		</p>
+		<p id="copy">
+			Copyright &copy; 2003-2019, <a class="samples" href="https://cksource.com/">CKSource</a> - Frederico
+			Knabben. All rights reserved.
+		</p>
+	</div>
+</body>
+</html>

+ 553 - 0
public/ckeditor/plugins/image2/dialogs/image2.js

@@ -0,0 +1,553 @@
+/**
+ * @license Copyright (c) 2003-2019, CKSource - Frederico Knabben. All rights reserved.
+ * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license
+ */
+
+/**
+ * @fileOverview Image plugin based on Widgets API
+ */
+
+'use strict';
+
+CKEDITOR.dialog.add( 'image2', function( editor ) {
+
+	// RegExp: 123, 123px, empty string ""
+	var regexGetSizeOrEmpty = /(^\s*(\d+)(px)?\s*$)|^$/i,
+
+		lockButtonId = CKEDITOR.tools.getNextId(),
+		resetButtonId = CKEDITOR.tools.getNextId(),
+
+		lang = editor.lang.image2,
+		commonLang = editor.lang.common,
+
+		lockResetStyle = 'margin-top:18px;width:40px;height:20px;',
+		lockResetHtml = new CKEDITOR.template(
+			'<div>' +
+				'<a href="javascript:void(0)" tabindex="-1" title="' + lang.lockRatio + '" class="cke_btn_locked" id="{lockButtonId}" role="checkbox">' +
+					'<span class="cke_icon"></span>' +
+					'<span class="cke_label">' + lang.lockRatio + '</span>' +
+				'</a>' +
+
+				'<a href="javascript:void(0)" tabindex="-1" title="' + lang.resetSize + '" class="cke_btn_reset" id="{resetButtonId}" role="button">' +
+					'<span class="cke_label">' + lang.resetSize + '</span>' +
+				'</a>' +
+			'</div>' ).output( {
+				lockButtonId: lockButtonId,
+				resetButtonId: resetButtonId
+			} ),
+
+		helpers = CKEDITOR.plugins.image2,
+
+		// Editor instance configuration.
+		config = editor.config,
+
+		hasFileBrowser = !!( config.filebrowserImageBrowseUrl || config.filebrowserBrowseUrl ),
+
+		// Content restrictions defined by the widget which
+		// impact on dialog structure and presence of fields.
+		features = editor.widgets.registered.image.features,
+
+		// Functions inherited from image2 plugin.
+		getNatural = helpers.getNatural,
+
+		// Global variables referring to the dialog's context.
+		doc, widget, image,
+
+		// Global variable referring to this dialog's image pre-loader.
+		preLoader,
+
+		// Global variables holding the original size of the image.
+		domWidth, domHeight,
+
+		// Global variables related to image pre-loading.
+		preLoadedWidth, preLoadedHeight, srcChanged,
+
+		// Global variables related to size locking.
+		lockRatio, userDefinedLock,
+
+		// Global variables referring to dialog fields and elements.
+		lockButton, resetButton, widthField, heightField,
+
+		natural;
+
+	// Validates dimension. Allowed values are:
+	// "123px", "123", "" (empty string)
+	function validateDimension() {
+		var match = this.getValue().match( regexGetSizeOrEmpty ),
+			isValid = !!( match && parseInt( match[ 1 ], 10 ) !== 0 );
+
+		if ( !isValid )
+			alert( commonLang[ 'invalidLength' ].replace( '%1', commonLang[ this.id ] ).replace( '%2', 'px' ) ); // jshint ignore:line
+
+		return isValid;
+	}
+
+	// Creates a function that pre-loads images. The callback function passes
+	// [image, width, height] or null if loading failed.
+	//
+	// @returns {Function}
+	function createPreLoader() {
+		var image = doc.createElement( 'img' ),
+			listeners = [];
+
+		function addListener( event, callback ) {
+			listeners.push( image.once( event, function( evt ) {
+				removeListeners();
+				callback( evt );
+			} ) );
+		}
+
+		function removeListeners() {
+			var l;
+
+			while ( ( l = listeners.pop() ) )
+				l.removeListener();
+		}
+
+		// @param {String} src.
+		// @param {Function} callback.
+		return function( src, callback, scope ) {
+			addListener( 'load', function() {
+				// Don't use image.$.(width|height) since it's buggy in IE9-10 (https://dev.ckeditor.com/ticket/11159)
+				var dimensions = getNatural( image );
+
+				callback.call( scope, image, dimensions.width, dimensions.height );
+			} );
+
+			addListener( 'error', function() {
+				callback( null );
+			} );
+
+			addListener( 'abort', function() {
+				callback( null );
+			} );
+
+			image.setAttribute( 'src',
+				( config.baseHref || '' ) + src + '?' + Math.random().toString( 16 ).substring( 2 ) );
+		};
+	}
+
+	// This function updates width and height fields once the
+	// "src" field is altered. Along with dimensions, also the
+	// dimensions lock is adjusted.
+	function onChangeSrc() {
+		var value = this.getValue();
+
+		toggleDimensions( false );
+
+		// Remember that src is different than default.
+		if ( value !== widget.data.src ) {
+			// Update dimensions of the image once it's preloaded.
+			preLoader( value, function( image, width, height ) {
+				// Re-enable width and height fields.
+				toggleDimensions( true );
+
+				// There was problem loading the image. Unlock ratio.
+				if ( !image )
+					return toggleLockRatio( false );
+
+				// Fill width field with the width of the new image.
+				widthField.setValue( editor.config.image2_prefillDimensions === false ? 0 : width );
+
+				// Fill height field with the height of the new image.
+				heightField.setValue( editor.config.image2_prefillDimensions === false ? 0 : height );
+
+				// Cache the new width and update initial cache (#1348).
+				preLoadedWidth = domWidth = width;
+
+				// Cache the new height and update initial cache (#1348).
+				preLoadedHeight = domHeight = height;
+
+				// Check for new lock value if image exist.
+				toggleLockRatio( helpers.checkHasNaturalRatio( image ) );
+			} );
+
+			srcChanged = true;
+		}
+
+		// Value is the same as in widget data but is was
+		// modified back in time. Roll back dimensions when restoring
+		// default src.
+		else if ( srcChanged ) {
+			// Re-enable width and height fields.
+			toggleDimensions( true );
+
+			// Restore width field with cached width.
+			widthField.setValue( domWidth );
+
+			// Restore height field with cached height.
+			heightField.setValue( domHeight );
+
+			// Src equals default one back again.
+			srcChanged = false;
+		}
+
+		// Value is the same as in widget data and it hadn't
+		// been modified.
+		else {
+			// Re-enable width and height fields.
+			toggleDimensions( true );
+		}
+	}
+
+	function onChangeDimension() {
+		// If ratio is un-locked, then we don't care what's next.
+		if ( !lockRatio )
+			return;
+
+		var value = this.getValue();
+
+		// No reason to auto-scale or unlock if the field is empty.
+		if ( !value )
+			return;
+
+		// If the value of the field is invalid (e.g. with %), unlock ratio.
+		if ( !value.match( regexGetSizeOrEmpty ) )
+			toggleLockRatio( false );
+
+		// No automatic re-scale when dimension is '0'.
+		if ( value === '0' )
+			return;
+
+		var isWidth = this.id == 'width',
+			// If dialog opened for the new image, domWidth and domHeight
+			// will be empty. Use dimensions from pre-loader in such case instead.
+			width = domWidth || preLoadedWidth,
+			height = domHeight || preLoadedHeight;
+
+		// If changing width, then auto-scale height.
+		if ( isWidth )
+			value = Math.round( height * ( value / width ) );
+
+		// If changing height, then auto-scale width.
+		else
+			value = Math.round( width * ( value / height ) );
+
+		// If the value is a number, apply it to the other field.
+		if ( !isNaN( value ) )
+			( isWidth ? heightField : widthField ).setValue( value );
+	}
+
+	// Set-up function for lock and reset buttons:
+	// 	* Adds lock and reset buttons to focusables. Check if button exist first
+	// 	  because it may be disabled e.g. due to ACF restrictions.
+	// 	* Register mouseover and mouseout event listeners for UI manipulations.
+	// 	* Register click event listeners for buttons.
+	function onLoadLockReset() {
+		var dialog = this.getDialog();
+
+		function setupMouseClasses( el ) {
+			el.on( 'mouseover', function() {
+				this.addClass( 'cke_btn_over' );
+			}, el );
+
+			el.on( 'mouseout', function() {
+				this.removeClass( 'cke_btn_over' );
+			}, el );
+		}
+
+		// Create references to lock and reset buttons for this dialog instance.
+		lockButton = doc.getById( lockButtonId );
+		resetButton = doc.getById( resetButtonId );
+
+		// Activate (Un)LockRatio button
+		if ( lockButton ) {
+			// Consider that there's an additional focusable field
+			// in the dialog when the "browse" button is visible.
+			dialog.addFocusable( lockButton, 4 + hasFileBrowser );
+
+			lockButton.on( 'click', function( evt ) {
+				toggleLockRatio();
+				evt.data && evt.data.preventDefault();
+			}, this.getDialog() );
+
+			setupMouseClasses( lockButton );
+		}
+
+		// Activate the reset size button.
+		if ( resetButton ) {
+			// Consider that there's an additional focusable field
+			// in the dialog when the "browse" button is visible.
+			dialog.addFocusable( resetButton, 5 + hasFileBrowser );
+
+			// Fills width and height fields with the original dimensions of the
+			// image (stored in widget#data since widget#init).
+			resetButton.on( 'click', function( evt ) {
+				// If there's a new image loaded, reset button should revert
+				// cached dimensions of pre-loaded DOM element.
+				if ( srcChanged ) {
+					widthField.setValue( preLoadedWidth );
+					heightField.setValue( preLoadedHeight );
+				}
+
+				// If the old image remains, reset button should revert
+				// dimensions as loaded when the dialog was first shown.
+				else {
+					widthField.setValue( domWidth );
+					heightField.setValue( domHeight );
+				}
+
+				evt.data && evt.data.preventDefault();
+			}, this );
+
+			setupMouseClasses( resetButton );
+		}
+	}
+
+	function toggleLockRatio( enable ) {
+		// No locking if there's no radio (i.e. due to ACF).
+		if ( !lockButton )
+			return;
+
+		if ( typeof enable == 'boolean' ) {
+			// If user explicitly wants to decide whether
+			// to lock or not, don't do anything.
+			if ( userDefinedLock )
+				return;
+
+			lockRatio = enable;
+		}
+
+		// Undefined. User changed lock value.
+		else {
+			var width = widthField.getValue(),
+				height;
+
+			userDefinedLock = true;
+			lockRatio = !lockRatio;
+
+			// Automatically adjust height to width to match
+			// the original ratio (based on dom- dimensions).
+			if ( lockRatio && width ) {
+				height = domHeight / domWidth * width;
+
+				if ( !isNaN( height ) )
+					heightField.setValue( Math.round( height ) );
+			}
+		}
+
+		lockButton[ lockRatio ? 'removeClass' : 'addClass' ]( 'cke_btn_unlocked' );
+		lockButton.setAttribute( 'aria-checked', lockRatio );
+
+		// Ratio button hc presentation - WHITE SQUARE / BLACK SQUARE
+		if ( CKEDITOR.env.hc ) {
+			var icon = lockButton.getChild( 0 );
+			icon.setHtml( lockRatio ? CKEDITOR.env.ie ? '\u25A0' : '\u25A3' : CKEDITOR.env.ie ? '\u25A1' : '\u25A2' );
+		}
+	}
+
+	function toggleDimensions( enable ) {
+		var method = enable ? 'enable' : 'disable';
+
+		widthField[ method ]();
+		heightField[ method ]();
+	}
+
+	var srcBoxChildren = [
+			{
+				id: 'src',
+				type: 'text',
+				label: commonLang.url,
+				onKeyup: onChangeSrc,
+				onChange: onChangeSrc,
+				setup: function( widget ) {
+					this.setValue( widget.data.src );
+				},
+				commit: function( widget ) {
+					widget.setData( 'src', this.getValue() );
+				},
+				validate: CKEDITOR.dialog.validate.notEmpty( lang.urlMissing )
+			}
+		];
+
+	// Render the "Browse" button on demand to avoid an "empty" (hidden child)
+	// space in dialog layout that distorts the UI.
+	if ( hasFileBrowser ) {
+		srcBoxChildren.push( {
+			type: 'button',
+			id: 'browse',
+			// v-align with the 'txtUrl' field.
+			// TODO: We need something better than a fixed size here.
+			style: 'display:inline-block;margin-top:14px;',
+			align: 'center',
+			label: editor.lang.common.browseServer,
+			hidden: true,
+			filebrowser: 'info:src'
+		} );
+	}
+
+	return {
+		title: lang.title,
+		minWidth: 250,
+		minHeight: 100,
+		onLoad: function() {
+			// Create a "global" reference to the document for this dialog instance.
+			doc = this._.element.getDocument();
+
+			// Create a pre-loader used for determining dimensions of new images.
+			preLoader = createPreLoader();
+		},
+		onShow: function() {
+			// Create a "global" reference to edited widget.
+			widget = this.getModel();
+
+			// Create a "global" reference to widget's image.
+			image = widget.parts.image;
+
+			// Reset global variables.
+			srcChanged = userDefinedLock = lockRatio = false;
+
+			// Natural dimensions of the image.
+			natural = getNatural( image );
+
+			// Get the natural width of the image.
+			preLoadedWidth = domWidth = natural.width;
+
+			// Get the natural height of the image.
+			preLoadedHeight = domHeight = natural.height;
+		},
+		contents: [
+			{
+				id: 'info',
+				label: lang.infoTab,
+				elements: [
+					{
+						type: 'vbox',
+						padding: 0,
+						children: [
+							{
+								type: 'hbox',
+								widths: [ '100%' ],
+								className: 'cke_dialog_image_url',
+								children: srcBoxChildren
+							}
+						]
+					},
+					{
+						id: 'alt',
+						type: 'text',
+						label: lang.alt,
+						setup: function( widget ) {
+							this.setValue( widget.data.alt );
+						},
+						commit: function( widget ) {
+							widget.setData( 'alt', this.getValue() );
+						},
+						validate: editor.config.image2_altRequired === true ? CKEDITOR.dialog.validate.notEmpty( lang.altMissing ) : null
+					},
+					{
+						type: 'hbox',
+						widths: [ '25%', '25%', '50%' ],
+						requiredContent: features.dimension.requiredContent,
+						children: [
+							{
+								type: 'text',
+								width: '45px',
+								id: 'width',
+								label: commonLang.width,
+								validate: validateDimension,
+								onKeyUp: onChangeDimension,
+								onLoad: function() {
+									widthField = this;
+								},
+								setup: function( widget ) {
+									this.setValue( widget.data.width );
+								},
+								commit: function( widget ) {
+									widget.setData( 'width', this.getValue() );
+								}
+							},
+							{
+								type: 'text',
+								id: 'height',
+								width: '45px',
+								label: commonLang.height,
+								validate: validateDimension,
+								onKeyUp: onChangeDimension,
+								onLoad: function() {
+									heightField = this;
+								},
+								setup: function( widget ) {
+									this.setValue( widget.data.height );
+								},
+								commit: function( widget ) {
+									widget.setData( 'height', this.getValue() );
+								}
+							},
+							{
+								id: 'lock',
+								type: 'html',
+								style: lockResetStyle,
+								onLoad: onLoadLockReset,
+								setup: function( widget ) {
+									toggleLockRatio( widget.data.lock );
+								},
+								commit: function( widget ) {
+									widget.setData( 'lock', lockRatio );
+								},
+								html: lockResetHtml
+							}
+						]
+					},
+					{
+						type: 'hbox',
+						id: 'alignment',
+						requiredContent: features.align.requiredContent,
+						children: [
+							{
+								id: 'align',
+								type: 'radio',
+								items: [
+									[ commonLang.alignNone, 'none' ],
+									[ commonLang.left, 'left' ],
+									[ commonLang.center, 'center' ],
+									[ commonLang.right, 'right' ]
+								],
+								label: commonLang.align,
+								setup: function( widget ) {
+									this.setValue( widget.data.align );
+								},
+								commit: function( widget ) {
+									widget.setData( 'align', this.getValue() );
+								}
+							}
+						]
+					},
+					{
+						id: 'hasCaption',
+						type: 'checkbox',
+						label: lang.captioned,
+						requiredContent: features.caption.requiredContent,
+						setup: function( widget ) {
+							this.setValue( widget.data.hasCaption );
+						},
+						commit: function( widget ) {
+							widget.setData( 'hasCaption', this.getValue() );
+						}
+					}
+				]
+			},
+			{
+				id: 'Upload',
+				hidden: true,
+				filebrowser: 'uploadButton',
+				label: lang.uploadTab,
+				elements: [
+					{
+						type: 'file',
+						id: 'upload',
+						label: lang.btnUpload,
+						style: 'height:40px'
+					},
+					{
+						type: 'fileButton',
+						id: 'uploadButton',
+						filebrowser: 'info:src',
+						label: lang.btnUpload,
+						'for': [ 'Upload', 'upload' ]
+					}
+				]
+			}
+		]
+	};
+} );

BIN
public/ckeditor/plugins/image2/icons/hidpi/image.png


BIN
public/ckeditor/plugins/image2/icons/image.png


+ 21 - 0
public/ckeditor/plugins/image2/lang/af.js

@@ -0,0 +1,21 @@
+/*
+Copyright (c) 2003-2019, CKSource - Frederico Knabben. All rights reserved.
+For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license
+*/
+CKEDITOR.plugins.setLang( 'image2', 'af', {
+	alt: 'Alternatiewe teks',
+	btnUpload: 'Stuur na bediener',
+	captioned: 'Captioned image', // MISSING
+	captionPlaceholder: 'Caption', // MISSING
+	infoTab: 'Afbeelding informasie',
+	lockRatio: 'Vaste proporsie',
+	menu: 'Afbeelding eienskappe',
+	pathName: 'image', // MISSING
+	pathNameCaption: 'caption', // MISSING
+	resetSize: 'Herstel grootte',
+	resizer: 'Click and drag to resize', // MISSING
+	title: 'Afbeelding eienskappe',
+	uploadTab: 'Oplaai',
+	urlMissing: 'Die URL na die afbeelding ontbreek.',
+	altMissing: 'Alternative text is missing.' // MISSING
+} );

+ 21 - 0
public/ckeditor/plugins/image2/lang/ar.js

@@ -0,0 +1,21 @@
+/*
+Copyright (c) 2003-2019, CKSource - Frederico Knabben. All rights reserved.
+For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license
+*/
+CKEDITOR.plugins.setLang( 'image2', 'ar', {
+	alt: 'عنوان الصورة',
+	btnUpload: 'أرسلها للخادم',
+	captioned: 'صورة ذات اسم',
+	captionPlaceholder: 'تسمية',
+	infoTab: 'معلومات الصورة',
+	lockRatio: 'تناسق الحجم',
+	menu: 'خصائص الصورة',
+	pathName: 'صورة',
+	pathNameCaption: 'تسمية',
+	resetSize: 'إستعادة الحجم الأصلي',
+	resizer: 'انقر ثم اسحب للتحجيم',
+	title: 'خصائص الصورة',
+	uploadTab: 'رفع',
+	urlMissing: 'عنوان مصدر الصورة مفقود',
+	altMissing: 'Alternative text is missing.' // MISSING
+} );

+ 21 - 0
public/ckeditor/plugins/image2/lang/az.js

@@ -0,0 +1,21 @@
+/*
+Copyright (c) 2003-2019, CKSource - Frederico Knabben. All rights reserved.
+For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license
+*/
+CKEDITOR.plugins.setLang( 'image2', 'az', {
+	alt: 'Alternativ mətn',
+	btnUpload: 'Serverə göndər',
+	captioned: 'Altyazı olan şəkil',
+	captionPlaceholder: 'Altyazı',
+	infoTab: 'Şəkil haqqında məlumat',
+	lockRatio: 'Ölçülərin nisbəti saxla',
+	menu: 'Şəklin seçimləri',
+	pathName: 'Şəkil',
+	pathNameCaption: 'Altyazı',
+	resetSize: 'Ölçüləri qaytar',
+	resizer: 'Ölçülər dəyişmək üçün tıklayın və aparın',
+	title: 'Şəklin seçimləri',
+	uploadTab: 'Serverə yüklə',
+	urlMissing: 'Şəklin ünvanı yanlışdır.',
+	altMissing: 'Alternativ mətn tapılmayıb'
+} );

+ 21 - 0
public/ckeditor/plugins/image2/lang/bg.js

@@ -0,0 +1,21 @@
+/*
+Copyright (c) 2003-2019, CKSource - Frederico Knabben. All rights reserved.
+For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license
+*/
+CKEDITOR.plugins.setLang( 'image2', 'bg', {
+	alt: 'Алтернативен текст',
+	btnUpload: 'Изпрати на сървъра',
+	captioned: 'Надписано изображение',
+	captionPlaceholder: 'Надпис',
+	infoTab: 'Изображение',
+	lockRatio: 'Заключване на съотношението',
+	menu: 'Настройки на изображението',
+	pathName: 'изображение',
+	pathNameCaption: 'надпис',
+	resetSize: 'Нулиране на размер',
+	resizer: 'Кликни и влачи, за да преоразмериш',
+	title: 'Настройки на изображението',
+	uploadTab: 'Качване',
+	urlMissing: 'URL адреса на изображението липсва.',
+	altMissing: 'Липсва алтернативен текст.'
+} );

+ 21 - 0
public/ckeditor/plugins/image2/lang/bn.js

@@ -0,0 +1,21 @@
+/*
+Copyright (c) 2003-2019, CKSource - Frederico Knabben. All rights reserved.
+For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license
+*/
+CKEDITOR.plugins.setLang( 'image2', 'bn', {
+	alt: 'বিকল্প টেক্সট',
+	btnUpload: 'ইহাকে সার্ভারে প্রেরন কর',
+	captioned: 'Captioned image', // MISSING
+	captionPlaceholder: 'Caption', // MISSING
+	infoTab: 'ছবির তথ্য',
+	lockRatio: 'অনুপাত লক কর',
+	menu: 'ছবির প্রোপার্টি',
+	pathName: 'image', // MISSING
+	pathNameCaption: 'caption', // MISSING
+	resetSize: 'সাইজ পূর্বাবস্থায় ফিরিয়ে দাও',
+	resizer: 'Click and drag to resize', // MISSING
+	title: 'ছবির প্রোপার্টি',
+	uploadTab: 'আপলোড',
+	urlMissing: 'Image source URL is missing.', // MISSING
+	altMissing: 'Alternative text is missing.' // MISSING
+} );

+ 21 - 0
public/ckeditor/plugins/image2/lang/bs.js

@@ -0,0 +1,21 @@
+/*
+Copyright (c) 2003-2019, CKSource - Frederico Knabben. All rights reserved.
+For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license
+*/
+CKEDITOR.plugins.setLang( 'image2', 'bs', {
+	alt: 'Tekst na slici',
+	btnUpload: 'Šalji na server',
+	captioned: 'Captioned image', // MISSING
+	captionPlaceholder: 'Caption', // MISSING
+	infoTab: 'Info slike',
+	lockRatio: 'Zakljuèaj odnos',
+	menu: 'Svojstva slike',
+	pathName: 'image', // MISSING
+	pathNameCaption: 'caption', // MISSING
+	resetSize: 'Resetuj dimenzije',
+	resizer: 'Click and drag to resize', // MISSING
+	title: 'Svojstva slike',
+	uploadTab: 'Šalji',
+	urlMissing: 'Image source URL is missing.', // MISSING
+	altMissing: 'Alternative text is missing.' // MISSING
+} );

+ 21 - 0
public/ckeditor/plugins/image2/lang/ca.js

@@ -0,0 +1,21 @@
+/*
+Copyright (c) 2003-2019, CKSource - Frederico Knabben. All rights reserved.
+For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license
+*/
+CKEDITOR.plugins.setLang( 'image2', 'ca', {
+	alt: 'Text alternatiu',
+	btnUpload: 'Envia-la al servidor',
+	captioned: 'Imatge amb subtítol',
+	captionPlaceholder: 'Títol',
+	infoTab: 'Informació de la imatge',
+	lockRatio: 'Bloqueja les proporcions',
+	menu: 'Propietats de la imatge',
+	pathName: 'imatge',
+	pathNameCaption: 'subtítol',
+	resetSize: 'Restaura la mida',
+	resizer: 'Clicar i arrossegar per redimensionar',
+	title: 'Propietats de la imatge',
+	uploadTab: 'Puja',
+	urlMissing: 'Falta la URL de la imatge.',
+	altMissing: 'Alternative text is missing.' // MISSING
+} );

+ 21 - 0
public/ckeditor/plugins/image2/lang/cs.js

@@ -0,0 +1,21 @@
+/*
+Copyright (c) 2003-2019, CKSource - Frederico Knabben. All rights reserved.
+For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license
+*/
+CKEDITOR.plugins.setLang( 'image2', 'cs', {
+	alt: 'Alternativní text',
+	btnUpload: 'Odeslat na server',
+	captioned: 'Obrázek s popisem',
+	captionPlaceholder: 'Popis',
+	infoTab: 'Informace o obrázku',
+	lockRatio: 'Zámek',
+	menu: 'Vlastnosti obrázku',
+	pathName: 'Obrázek',
+	pathNameCaption: 'Popis',
+	resetSize: 'Původní velikost',
+	resizer: 'Klepněte a táhněte pro změnu velikosti',
+	title: 'Vlastnosti obrázku',
+	uploadTab: 'Odeslat',
+	urlMissing: 'Zadané URL zdroje obrázku nebylo nalezeno.',
+	altMissing: 'Alternativní text chybí.'
+} );

+ 21 - 0
public/ckeditor/plugins/image2/lang/cy.js

@@ -0,0 +1,21 @@
+/*
+Copyright (c) 2003-2019, CKSource - Frederico Knabben. All rights reserved.
+For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license
+*/
+CKEDITOR.plugins.setLang( 'image2', 'cy', {
+	alt: 'Testun Amgen',
+	btnUpload: 'Anfon i\'r Gweinydd',
+	captioned: 'Delwedd â phennawd',
+	captionPlaceholder: 'Caption', // MISSING
+	infoTab: 'Gwyb Delwedd',
+	lockRatio: 'Cloi Cymhareb',
+	menu: 'Priodweddau Delwedd',
+	pathName: 'delwedd',
+	pathNameCaption: 'pennawd',
+	resetSize: 'Ailosod Maint',
+	resizer: 'Clicio a llusgo i ail-meintio',
+	title: 'Priodweddau Delwedd',
+	uploadTab: 'Lanlwytho',
+	urlMissing: 'URL gwreiddiol y ddelwedd ar goll.',
+	altMissing: 'Alternative text is missing.' // MISSING
+} );

+ 21 - 0
public/ckeditor/plugins/image2/lang/da.js

@@ -0,0 +1,21 @@
+/*
+Copyright (c) 2003-2019, CKSource - Frederico Knabben. All rights reserved.
+For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license
+*/
+CKEDITOR.plugins.setLang( 'image2', 'da', {
+	alt: 'Alternativ tekst',
+	btnUpload: 'Upload fil til serveren',
+	captioned: 'Tekstet billede',
+	captionPlaceholder: 'Tekst',
+	infoTab: 'Generelt',
+	lockRatio: 'Lås størrelsesforhold',
+	menu: 'Egenskaber for billede',
+	pathName: 'billede',
+	pathNameCaption: 'tekst',
+	resetSize: 'Nulstil størrelse',
+	resizer: 'Klik og træk for at ændre størrelsen',
+	title: 'Egenskaber for billede',
+	uploadTab: 'Upload',
+	urlMissing: 'Kilde på billed-URL mangler',
+	altMissing: 'Alternative text is missing.' // MISSING
+} );

+ 21 - 0
public/ckeditor/plugins/image2/lang/de-ch.js

@@ -0,0 +1,21 @@
+/*
+Copyright (c) 2003-2019, CKSource - Frederico Knabben. All rights reserved.
+For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license
+*/
+CKEDITOR.plugins.setLang( 'image2', 'de-ch', {
+	alt: 'Alternativer Text',
+	btnUpload: 'Zum Server senden',
+	captioned: 'Bild mit Überschrift',
+	captionPlaceholder: 'Überschrift',
+	infoTab: 'Bildinfo',
+	lockRatio: 'Größenverhältnis beibehalten',
+	menu: 'Bildeigenschaften',
+	pathName: 'Bild',
+	pathNameCaption: 'Überschrift',
+	resetSize: 'Grösse zurücksetzen',
+	resizer: 'Zum Vergrössern auswählen und ziehen',
+	title: 'Bild-Eigenschaften',
+	uploadTab: 'Hochladen',
+	urlMissing: 'Bildquellen-URL fehlt.',
+	altMissing: 'Alternative text is missing.' // MISSING
+} );

+ 21 - 0
public/ckeditor/plugins/image2/lang/de.js

@@ -0,0 +1,21 @@
+/*
+Copyright (c) 2003-2019, CKSource - Frederico Knabben. All rights reserved.
+For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license
+*/
+CKEDITOR.plugins.setLang( 'image2', 'de', {
+	alt: 'Alternativer Text',
+	btnUpload: 'Zum Server senden',
+	captioned: 'Bild mit Überschrift',
+	captionPlaceholder: 'Überschrift',
+	infoTab: 'Bildinfo',
+	lockRatio: 'Größenverhältnis beibehalten',
+	menu: 'Bildeigenschaften',
+	pathName: 'Bild',
+	pathNameCaption: 'Überschrift',
+	resetSize: 'Größe zurücksetzen',
+	resizer: 'Zum Vergrößern auswählen und ziehen',
+	title: 'Bild-Eigenschaften',
+	uploadTab: 'Hochladen',
+	urlMissing: 'Bildquellen-URL fehlt.',
+	altMissing: 'Alternativer Text fehlt.'
+} );

+ 21 - 0
public/ckeditor/plugins/image2/lang/el.js

@@ -0,0 +1,21 @@
+/*
+Copyright (c) 2003-2019, CKSource - Frederico Knabben. All rights reserved.
+For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license
+*/
+CKEDITOR.plugins.setLang( 'image2', 'el', {
+	alt: 'Εναλλακτικό Κείμενο',
+	btnUpload: 'Αποστολή στον Διακομιστή',
+	captioned: 'Εικόνα με λεζάντα',
+	captionPlaceholder: 'Λεζάντα',
+	infoTab: 'Πληροφορίες Εικόνας',
+	lockRatio: 'Κλείδωμα Αναλογίας',
+	menu: 'Ιδιότητες Εικόνας',
+	pathName: 'εικόνα',
+	pathNameCaption: 'λεζάντα',
+	resetSize: 'Επαναφορά Αρχικού Μεγέθους',
+	resizer: 'Κάνετε κλικ και σύρετε το ποντίκι για να αλλάξετε το μέγεθος',
+	title: 'Ιδιότητες Εικόνας',
+	uploadTab: 'Αποστολή',
+	urlMissing: 'Λείπει το πηγαίο URL της εικόνας.',
+	altMissing: 'Alternative text is missing.' // MISSING
+} );

+ 21 - 0
public/ckeditor/plugins/image2/lang/en-au.js

@@ -0,0 +1,21 @@
+/*
+Copyright (c) 2003-2019, CKSource - Frederico Knabben. All rights reserved.
+For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license
+*/
+CKEDITOR.plugins.setLang( 'image2', 'en-au', {
+	alt: 'Alternative Text',
+	btnUpload: 'Send it to the Server',
+	captioned: 'Captioned image',
+	captionPlaceholder: 'Caption',
+	infoTab: 'Image Info',
+	lockRatio: 'Lock Ratio',
+	menu: 'Image Properties',
+	pathName: 'image',
+	pathNameCaption: 'caption',
+	resetSize: 'Reset Size',
+	resizer: 'Click and drag to resize',
+	title: 'Image Properties',
+	uploadTab: 'Upload',
+	urlMissing: 'Image source URL is missing.',
+	altMissing: 'Alternative text is missing.'
+} );

+ 21 - 0
public/ckeditor/plugins/image2/lang/en-ca.js

@@ -0,0 +1,21 @@
+/*
+Copyright (c) 2003-2019, CKSource - Frederico Knabben. All rights reserved.
+For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license
+*/
+CKEDITOR.plugins.setLang( 'image2', 'en-ca', {
+	alt: 'Alternative Text',
+	btnUpload: 'Send it to the Server',
+	captioned: 'Captioned image', // MISSING
+	captionPlaceholder: 'Caption', // MISSING
+	infoTab: 'Image Info',
+	lockRatio: 'Lock Ratio',
+	menu: 'Image Properties',
+	pathName: 'image', // MISSING
+	pathNameCaption: 'caption', // MISSING
+	resetSize: 'Reset Size',
+	resizer: 'Click and drag to resize', // MISSING
+	title: 'Image Properties',
+	uploadTab: 'Upload',
+	urlMissing: 'Image source URL is missing.', // MISSING
+	altMissing: 'Alternative text is missing.' // MISSING
+} );

+ 21 - 0
public/ckeditor/plugins/image2/lang/en-gb.js

@@ -0,0 +1,21 @@
+/*
+Copyright (c) 2003-2019, CKSource - Frederico Knabben. All rights reserved.
+For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license
+*/
+CKEDITOR.plugins.setLang( 'image2', 'en-gb', {
+	alt: 'Alternative Text',
+	btnUpload: 'Send it to the Server',
+	captioned: 'Captioned image',
+	captionPlaceholder: 'Caption',
+	infoTab: 'Image Info',
+	lockRatio: 'Lock Ratio',
+	menu: 'Image Properties',
+	pathName: 'image',
+	pathNameCaption: 'caption',
+	resetSize: 'Reset Size',
+	resizer: 'Click and drag to resize',
+	title: 'Image Properties',
+	uploadTab: 'Upload',
+	urlMissing: 'Image source URL is missing.',
+	altMissing: 'Alternative text is missing.' // MISSING
+} );

+ 21 - 0
public/ckeditor/plugins/image2/lang/en.js

@@ -0,0 +1,21 @@
+/*
+Copyright (c) 2003-2019, CKSource - Frederico Knabben. All rights reserved.
+For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license
+*/
+CKEDITOR.plugins.setLang( 'image2', 'en', {
+	alt: 'Alternative Text',
+	btnUpload: 'Send it to the Server',
+	captioned: 'Captioned image',
+	captionPlaceholder: 'Caption',
+	infoTab: 'Image Info',
+	lockRatio: 'Lock Ratio',
+	menu: 'Image Properties',
+	pathName: 'image',
+	pathNameCaption: 'caption',
+	resetSize: 'Reset Size',
+	resizer: 'Click and drag to resize',
+	title: 'Image Properties',
+	uploadTab: 'Upload',
+	urlMissing: 'Image source URL is missing.',
+	altMissing: 'Alternative text is missing.'
+} );

+ 21 - 0
public/ckeditor/plugins/image2/lang/eo.js

@@ -0,0 +1,21 @@
+/*
+Copyright (c) 2003-2019, CKSource - Frederico Knabben. All rights reserved.
+For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license
+*/
+CKEDITOR.plugins.setLang( 'image2', 'eo', {
+	alt: 'Anstataŭiga Teksto',
+	btnUpload: 'Sendu al Servilo',
+	captioned: 'Bildo kun apudskribo',
+	captionPlaceholder: 'Apudskribo',
+	infoTab: 'Informoj pri Bildo',
+	lockRatio: 'Konservi Proporcion',
+	menu: 'Atributoj de Bildo',
+	pathName: 'bildo',
+	pathNameCaption: 'apudskribo',
+	resetSize: 'Origina Grando',
+	resizer: 'Kliki kaj treni por ŝanĝi la grandon',
+	title: 'Atributoj de Bildo',
+	uploadTab: 'Alŝuti',
+	urlMissing: 'La fontretadreso de la bildo mankas.',
+	altMissing: 'Alternativa teksto mankas.'
+} );

+ 21 - 0
public/ckeditor/plugins/image2/lang/es-mx.js

@@ -0,0 +1,21 @@
+/*
+Copyright (c) 2003-2019, CKSource - Frederico Knabben. All rights reserved.
+For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license
+*/
+CKEDITOR.plugins.setLang( 'image2', 'es-mx', {
+	alt: 'Texto alternativo',
+	btnUpload: 'Enviar al servidor',
+	captioned: 'Imagen subtitulada',
+	captionPlaceholder: 'Subtítulo',
+	infoTab: 'Información de la imagen',
+	lockRatio: 'Bloquear aspecto',
+	menu: 'Propiedades de la imagen',
+	pathName: 'imagen',
+	pathNameCaption: 'subtítulo',
+	resetSize: 'Reiniciar tamaño',
+	resizer: 'Presiona y arrastra para redimensionar',
+	title: 'Propiedades de imagen',
+	uploadTab: 'Cargar',
+	urlMissing: 'Falta la URL de origen de la imagen.',
+	altMissing: 'Falta texto alternativo.'
+} );

+ 21 - 0
public/ckeditor/plugins/image2/lang/es.js

@@ -0,0 +1,21 @@
+/*
+Copyright (c) 2003-2019, CKSource - Frederico Knabben. All rights reserved.
+For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license
+*/
+CKEDITOR.plugins.setLang( 'image2', 'es', {
+	alt: 'Texto Alternativo',
+	btnUpload: 'Enviar al Servidor',
+	captioned: 'Imagen subtitulada',
+	captionPlaceholder: 'Leyenda',
+	infoTab: 'Información de Imagen',
+	lockRatio: 'Proporcional',
+	menu: 'Propiedades de Imagen',
+	pathName: 'image',
+	pathNameCaption: 'subtítulo',
+	resetSize: 'Tamaño Original',
+	resizer: 'Dar clic y arrastrar para cambiar tamaño',
+	title: 'Propiedades de Imagen',
+	uploadTab: 'Cargar',
+	urlMissing: 'Debe indicar la URL de la imagen.',
+	altMissing: 'Alternative text is missing.' // MISSING
+} );

+ 21 - 0
public/ckeditor/plugins/image2/lang/et.js

@@ -0,0 +1,21 @@
+/*
+Copyright (c) 2003-2019, CKSource - Frederico Knabben. All rights reserved.
+For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license
+*/
+CKEDITOR.plugins.setLang( 'image2', 'et', {
+	alt: 'Alternatiivne tekst',
+	btnUpload: 'Saada serverisse',
+	captioned: 'Pealkirjaga pilt',
+	captionPlaceholder: 'Pealkiri',
+	infoTab: 'Pildi info',
+	lockRatio: 'Lukusta kuvasuhe',
+	menu: 'Pildi omadused',
+	pathName: 'pilt',
+	pathNameCaption: 'pealkiri',
+	resetSize: 'Lähtesta suurus',
+	resizer: 'Suuruse muutmiseks klõpsa ja lohista',
+	title: 'Pildi omadused',
+	uploadTab: 'Lae üles',
+	urlMissing: 'Pildi lähte-URL on puudu.',
+	altMissing: 'Alternatiivtekst puudub.'
+} );

+ 21 - 0
public/ckeditor/plugins/image2/lang/eu.js

@@ -0,0 +1,21 @@
+/*
+Copyright (c) 2003-2019, CKSource - Frederico Knabben. All rights reserved.
+For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license
+*/
+CKEDITOR.plugins.setLang( 'image2', 'eu', {
+	alt: 'Ordezko testua',
+	btnUpload: 'Bidali zerbitzarira',
+	captioned: 'Argazki oina',
+	captionPlaceholder: 'Argazki oina',
+	infoTab: 'Irudiaren informazioa',
+	lockRatio: 'Blokeatu erlazioa',
+	menu: 'Irudiaren propietateak',
+	pathName: 'Irudia',
+	pathNameCaption: 'Argazki oina',
+	resetSize: 'Berrezarri tamaina',
+	resizer: 'Klikatu eta arrastatu tamainaz aldatzeko',
+	title: 'Irudiaren propietateak',
+	uploadTab: 'Kargatu',
+	urlMissing: 'Irudiaren iturburuaren URLa falta da.',
+	altMissing: 'Ordezko testua falta da.'
+} );

+ 21 - 0
public/ckeditor/plugins/image2/lang/fa.js

@@ -0,0 +1,21 @@
+/*
+Copyright (c) 2003-2019, CKSource - Frederico Knabben. All rights reserved.
+For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license
+*/
+CKEDITOR.plugins.setLang( 'image2', 'fa', {
+	alt: 'متن جایگزین',
+	btnUpload: 'به سرور بفرست',
+	captioned: 'تصویر زیرنویس شده',
+	captionPlaceholder: 'عنوان',
+	infoTab: 'اطلاعات تصویر',
+	lockRatio: 'قفل کردن نسبت',
+	menu: 'ویژگی​های تصویر',
+	pathName: 'تصویر',
+	pathNameCaption: 'عنوان',
+	resetSize: 'بازنشانی اندازه',
+	resizer: 'کلیک و کشیدن برای تغییر اندازه',
+	title: 'ویژگی​های تصویر',
+	uploadTab: 'بالاگذاری',
+	urlMissing: 'آدرس URL اصلی تصویر یافت نشد.',
+	altMissing: 'متن جایگزین یافت نشد.'
+} );

+ 21 - 0
public/ckeditor/plugins/image2/lang/fi.js

@@ -0,0 +1,21 @@
+/*
+Copyright (c) 2003-2019, CKSource - Frederico Knabben. All rights reserved.
+For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license
+*/
+CKEDITOR.plugins.setLang( 'image2', 'fi', {
+	alt: 'Vaihtoehtoinen teksti',
+	btnUpload: 'Lähetä palvelimelle',
+	captioned: 'Kuva kuvatekstillä',
+	captionPlaceholder: 'Kuvateksti',
+	infoTab: 'Kuvan tiedot',
+	lockRatio: 'Lukitse suhteet',
+	menu: 'Kuvan ominaisuudet',
+	pathName: 'kuva',
+	pathNameCaption: 'kuvateksti',
+	resetSize: 'Alkuperäinen koko',
+	resizer: 'Klikkaa ja raahaa muuttaaksesi kokoa',
+	title: 'Kuvan ominaisuudet',
+	uploadTab: 'Lisää tiedosto',
+	urlMissing: 'Kuvan lähdeosoite puuttuu.',
+	altMissing: 'Alternative text is missing.' // MISSING
+} );

+ 21 - 0
public/ckeditor/plugins/image2/lang/fo.js

@@ -0,0 +1,21 @@
+/*
+Copyright (c) 2003-2019, CKSource - Frederico Knabben. All rights reserved.
+For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license
+*/
+CKEDITOR.plugins.setLang( 'image2', 'fo', {
+	alt: 'Alternativur tekstur',
+	btnUpload: 'Send til ambætaran',
+	captioned: 'Captioned image', // MISSING
+	captionPlaceholder: 'Caption', // MISSING
+	infoTab: 'Myndaupplýsingar',
+	lockRatio: 'Læs lutfallið',
+	menu: 'Myndaeginleikar',
+	pathName: 'image', // MISSING
+	pathNameCaption: 'caption', // MISSING
+	resetSize: 'Upprunastødd',
+	resizer: 'Click and drag to resize', // MISSING
+	title: 'Myndaeginleikar',
+	uploadTab: 'Send til ambætaran',
+	urlMissing: 'URL til mynd manglar.',
+	altMissing: 'Alternative text is missing.' // MISSING
+} );

+ 21 - 0
public/ckeditor/plugins/image2/lang/fr-ca.js

@@ -0,0 +1,21 @@
+/*
+Copyright (c) 2003-2019, CKSource - Frederico Knabben. All rights reserved.
+For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license
+*/
+CKEDITOR.plugins.setLang( 'image2', 'fr-ca', {
+	alt: 'Texte alternatif',
+	btnUpload: 'Envoyer sur le serveur',
+	captioned: 'Captioned image', // MISSING
+	captionPlaceholder: 'Caption', // MISSING
+	infoTab: 'Informations sur l\'image2',
+	lockRatio: 'Verrouiller les proportions',
+	menu: 'Propriétés de l\'image2',
+	pathName: 'image', // MISSING
+	pathNameCaption: 'caption', // MISSING
+	resetSize: 'Taille originale',
+	resizer: 'Click and drag to resize', // MISSING
+	title: 'Propriétés de l\'image2',
+	uploadTab: 'Téléverser',
+	urlMissing: 'L\'URL de la source de l\'image est manquant.',
+	altMissing: 'Alternative text is missing.' // MISSING
+} );

+ 21 - 0
public/ckeditor/plugins/image2/lang/fr.js

@@ -0,0 +1,21 @@
+/*
+Copyright (c) 2003-2019, CKSource - Frederico Knabben. All rights reserved.
+For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license
+*/
+CKEDITOR.plugins.setLang( 'image2', 'fr', {
+	alt: 'Texte alternatif',
+	btnUpload: 'Envoyer sur le serveur',
+	captioned: 'Image légendée',
+	captionPlaceholder: 'Légende',
+	infoTab: 'Informations sur l\'image',
+	lockRatio: 'Conserver les proportions',
+	menu: 'Propriétés de l\'image',
+	pathName: 'image',
+	pathNameCaption: 'légende',
+	resetSize: 'Réinitialiser la taille',
+	resizer: 'Cliquer et glisser pour redimensionner',
+	title: 'Propriétés de l\'image',
+	uploadTab: 'Téléverser',
+	urlMissing: 'L\'URL source de l\'image est manquante.',
+	altMissing: 'Vous n\'avez pas indiqué de texte de remplacement.'
+} );

+ 21 - 0
public/ckeditor/plugins/image2/lang/gl.js

@@ -0,0 +1,21 @@
+/*
+Copyright (c) 2003-2019, CKSource - Frederico Knabben. All rights reserved.
+For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license
+*/
+CKEDITOR.plugins.setLang( 'image2', 'gl', {
+	alt: 'Texto alternativo',
+	btnUpload: 'Enviar ao servidor',
+	captioned: 'Imaxe con lenda',
+	captionPlaceholder: 'Lenda',
+	infoTab: 'Información da imaxe',
+	lockRatio: 'Proporcional',
+	menu: 'Propiedades da imaxe',
+	pathName: 'Imaxe',
+	pathNameCaption: 'lenda',
+	resetSize: 'Tamaño orixinal',
+	resizer: 'Prema e arrastre para axustar o tamaño',
+	title: 'Propiedades da imaxe',
+	uploadTab: 'Cargar',
+	urlMissing: 'Non se atopa o URL da imaxe.',
+	altMissing: 'Non foi posíbel atopar o texto alternativo.'
+} );

+ 21 - 0
public/ckeditor/plugins/image2/lang/gu.js

@@ -0,0 +1,21 @@
+/*
+Copyright (c) 2003-2019, CKSource - Frederico Knabben. All rights reserved.
+For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license
+*/
+CKEDITOR.plugins.setLang( 'image2', 'gu', {
+	alt: 'ઑલ્ટર્નટ ટેક્સ્ટ',
+	btnUpload: 'આ સર્વરને મોકલવું',
+	captioned: 'Captioned image', // MISSING
+	captionPlaceholder: 'Caption', // MISSING
+	infoTab: 'ચિત્ર ની જાણકારી',
+	lockRatio: 'લૉક ગુણોત્તર',
+	menu: 'ચિત્રના ગુણ',
+	pathName: 'image', // MISSING
+	pathNameCaption: 'caption', // MISSING
+	resetSize: 'રીસેટ સાઇઝ',
+	resizer: 'Click and drag to resize', // MISSING
+	title: 'ચિત્રના ગુણ',
+	uploadTab: 'અપલોડ',
+	urlMissing: 'ઈમેજની મૂળ URL છે નહી.',
+	altMissing: 'Alternative text is missing.' // MISSING
+} );

+ 21 - 0
public/ckeditor/plugins/image2/lang/he.js

@@ -0,0 +1,21 @@
+/*
+Copyright (c) 2003-2019, CKSource - Frederico Knabben. All rights reserved.
+For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license
+*/
+CKEDITOR.plugins.setLang( 'image2', 'he', {
+	alt: 'טקסט חלופי',
+	btnUpload: 'שליחה לשרת',
+	captioned: 'כותרת תמונה',
+	captionPlaceholder: 'כותרת',
+	infoTab: 'מידע על התמונה',
+	lockRatio: 'נעילת היחס',
+	menu: 'תכונות התמונה',
+	pathName: 'תמונה',
+	pathNameCaption: 'כותרת',
+	resetSize: 'איפוס הגודל',
+	resizer: 'לחץ וגרור לשינוי הגודל',
+	title: 'מאפייני התמונה',
+	uploadTab: 'העלאה',
+	urlMissing: 'כתובת התמונה חסרה.',
+	altMissing: 'Alternative text is missing.' // MISSING
+} );

+ 21 - 0
public/ckeditor/plugins/image2/lang/hi.js

@@ -0,0 +1,21 @@
+/*
+Copyright (c) 2003-2019, CKSource - Frederico Knabben. All rights reserved.
+For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license
+*/
+CKEDITOR.plugins.setLang( 'image2', 'hi', {
+	alt: 'वैकल्पिक टेक्स्ट',
+	btnUpload: 'इसे सर्वर को भेजें',
+	captioned: 'Captioned image', // MISSING
+	captionPlaceholder: 'Caption', // MISSING
+	infoTab: 'तस्वीर की जानकारी',
+	lockRatio: 'लॉक अनुपात',
+	menu: 'तस्वीर प्रॉपर्टीज़',
+	pathName: 'image', // MISSING
+	pathNameCaption: 'caption', // MISSING
+	resetSize: 'रीसॅट साइज़',
+	resizer: 'Click and drag to resize', // MISSING
+	title: 'तस्वीर प्रॉपर्टीज़',
+	uploadTab: 'अपलोड',
+	urlMissing: 'Image source URL is missing.', // MISSING
+	altMissing: 'Alternative text is missing.' // MISSING
+} );

+ 21 - 0
public/ckeditor/plugins/image2/lang/hr.js

@@ -0,0 +1,21 @@
+/*
+Copyright (c) 2003-2019, CKSource - Frederico Knabben. All rights reserved.
+For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license
+*/
+CKEDITOR.plugins.setLang( 'image2', 'hr', {
+	alt: 'Alternativni tekst',
+	btnUpload: 'Pošalji na server',
+	captioned: 'Titl slike',
+	captionPlaceholder: 'Titl',
+	infoTab: 'Info slike',
+	lockRatio: 'Zaključaj odnos',
+	menu: 'Svojstva slika',
+	pathName: 'slika',
+	pathNameCaption: 'titl',
+	resetSize: 'Obriši veličinu',
+	resizer: 'Odaberi i povuci za promjenu veličine',
+	title: 'Svojstva slika',
+	uploadTab: 'Pošalji',
+	urlMissing: 'Nedostaje URL slike.',
+	altMissing: 'Nedostaje alternativni tekst.'
+} );

+ 21 - 0
public/ckeditor/plugins/image2/lang/hu.js

@@ -0,0 +1,21 @@
+/*
+Copyright (c) 2003-2019, CKSource - Frederico Knabben. All rights reserved.
+For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license
+*/
+CKEDITOR.plugins.setLang( 'image2', 'hu', {
+	alt: 'Alternatív szöveg',
+	btnUpload: 'Küldés a szerverre',
+	captioned: 'Feliratozott kép',
+	captionPlaceholder: 'Képfelirat',
+	infoTab: 'Alaptulajdonságok',
+	lockRatio: 'Arány megtartása',
+	menu: 'Kép tulajdonságai',
+	pathName: 'kép',
+	pathNameCaption: 'felirat',
+	resetSize: 'Eredeti méret',
+	resizer: 'Kattintson és húzza az átméretezéshez',
+	title: 'Kép tulajdonságai',
+	uploadTab: 'Feltöltés',
+	urlMissing: 'Hiányzik a kép URL-je',
+	altMissing: 'Az alternatív szöveg hiányzik.'
+} );

+ 21 - 0
public/ckeditor/plugins/image2/lang/id.js

@@ -0,0 +1,21 @@
+/*
+Copyright (c) 2003-2019, CKSource - Frederico Knabben. All rights reserved.
+For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license
+*/
+CKEDITOR.plugins.setLang( 'image2', 'id', {
+	alt: 'Teks alternatif',
+	btnUpload: 'Kirim ke Server',
+	captioned: 'Captioned image', // MISSING
+	captionPlaceholder: 'Caption', // MISSING
+	infoTab: 'Info Gambar',
+	lockRatio: 'Lock Ratio', // MISSING
+	menu: 'Image Properties', // MISSING
+	pathName: 'image', // MISSING
+	pathNameCaption: 'caption', // MISSING
+	resetSize: 'Atur Ulang Ukuran',
+	resizer: 'Click and drag to resize', // MISSING
+	title: 'Image Properties', // MISSING
+	uploadTab: 'Unggah',
+	urlMissing: 'Image source URL is missing.', // MISSING
+	altMissing: 'Alternative text is missing.' // MISSING
+} );

+ 21 - 0
public/ckeditor/plugins/image2/lang/is.js

@@ -0,0 +1,21 @@
+/*
+Copyright (c) 2003-2019, CKSource - Frederico Knabben. All rights reserved.
+For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license
+*/
+CKEDITOR.plugins.setLang( 'image2', 'is', {
+	alt: 'Baklægur texti',
+	btnUpload: 'Hlaða upp',
+	captioned: 'Captioned image', // MISSING
+	captionPlaceholder: 'Caption', // MISSING
+	infoTab: 'Almennt',
+	lockRatio: 'Festa stærðarhlutfall',
+	menu: 'Eigindi myndar',
+	pathName: 'image', // MISSING
+	pathNameCaption: 'caption', // MISSING
+	resetSize: 'Reikna stærð',
+	resizer: 'Click and drag to resize', // MISSING
+	title: 'Eigindi myndar',
+	uploadTab: 'Senda upp',
+	urlMissing: 'Image source URL is missing.', // MISSING
+	altMissing: 'Alternative text is missing.' // MISSING
+} );

+ 21 - 0
public/ckeditor/plugins/image2/lang/it.js

@@ -0,0 +1,21 @@
+/*
+Copyright (c) 2003-2019, CKSource - Frederico Knabben. All rights reserved.
+For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license
+*/
+CKEDITOR.plugins.setLang( 'image2', 'it', {
+	alt: 'Testo alternativo',
+	btnUpload: 'Invia al server',
+	captioned: 'Immagine con didascalia',
+	captionPlaceholder: 'Didascalia',
+	infoTab: 'Informazioni immagine',
+	lockRatio: 'Blocca rapporto',
+	menu: 'Proprietà immagine',
+	pathName: 'immagine',
+	pathNameCaption: 'didascalia',
+	resetSize: 'Reimposta dimensione',
+	resizer: 'Fare clic e trascinare per ridimensionare',
+	title: 'Proprietà immagine',
+	uploadTab: 'Carica',
+	urlMissing: 'Manca l\'URL dell\'immagine.',
+	altMissing: 'Testo alternativo mancante.'
+} );

+ 21 - 0
public/ckeditor/plugins/image2/lang/ja.js

@@ -0,0 +1,21 @@
+/*
+Copyright (c) 2003-2019, CKSource - Frederico Knabben. All rights reserved.
+For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license
+*/
+CKEDITOR.plugins.setLang( 'image2', 'ja', {
+	alt: '代替テキスト',
+	btnUpload: 'サーバーに送信',
+	captioned: 'キャプションを付ける',
+	captionPlaceholder: 'キャプション',
+	infoTab: '画像情報',
+	lockRatio: '比率を固定',
+	menu: '画像のプロパティ',
+	pathName: 'image',
+	pathNameCaption: 'caption',
+	resetSize: 'サイズをリセット',
+	resizer: 'ドラッグしてリサイズ',
+	title: '画像のプロパティ',
+	uploadTab: 'アップロード',
+	urlMissing: '画像のURLを入力してください。',
+	altMissing: '代替テキストを入力してください。'
+} );

+ 21 - 0
public/ckeditor/plugins/image2/lang/ka.js

@@ -0,0 +1,21 @@
+/*
+Copyright (c) 2003-2019, CKSource - Frederico Knabben. All rights reserved.
+For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license
+*/
+CKEDITOR.plugins.setLang( 'image2', 'ka', {
+	alt: 'სანაცვლო ტექსტი',
+	btnUpload: 'სერვერისთვის გაგზავნა',
+	captioned: 'Captioned image', // MISSING
+	captionPlaceholder: 'Caption', // MISSING
+	infoTab: 'სურათის ინფორმცია',
+	lockRatio: 'პროპორციის შენარჩუნება',
+	menu: 'სურათის პარამეტრები',
+	pathName: 'image', // MISSING
+	pathNameCaption: 'caption', // MISSING
+	resetSize: 'ზომის დაბრუნება',
+	resizer: 'Click and drag to resize', // MISSING
+	title: 'სურათის პარამეტრები',
+	uploadTab: 'აქაჩვა',
+	urlMissing: 'სურათის URL არაა შევსებული.',
+	altMissing: 'Alternative text is missing.' // MISSING
+} );

+ 21 - 0
public/ckeditor/plugins/image2/lang/km.js

@@ -0,0 +1,21 @@
+/*
+Copyright (c) 2003-2019, CKSource - Frederico Knabben. All rights reserved.
+For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license
+*/
+CKEDITOR.plugins.setLang( 'image2', 'km', {
+	alt: 'អត្ថបទជំនួស',
+	btnUpload: 'បញ្ជូនទៅកាន់ម៉ាស៊ីនផ្តល់សេវា',
+	captioned: 'រូប​ដែល​មាន​ចំណង​ជើង',
+	captionPlaceholder: 'Caption', // MISSING
+	infoTab: 'ពត៌មានអំពីរូបភាព',
+	lockRatio: 'ចាក់​សោ​ផល​ធៀប',
+	menu: 'លក្ខណៈ​សម្បត្តិ​រូប​ភាព',
+	pathName: 'រូបភាព',
+	pathNameCaption: 'ចំណងជើង',
+	resetSize: 'កំណត់ទំហំឡើងវិញ',
+	resizer: 'ចុច​ហើយ​ទាញ​ដើម្បី​ប្ដូរ​ទំហំ',
+	title: 'លក្ខណៈ​សម្បត្តិ​រូប​ភាប',
+	uploadTab: 'ផ្ទុក​ឡើង',
+	urlMissing: 'ខ្វះ URL ប្រភព​រូប​ភាព។',
+	altMissing: 'Alternative text is missing.' // MISSING
+} );

+ 21 - 0
public/ckeditor/plugins/image2/lang/ko.js

@@ -0,0 +1,21 @@
+/*
+Copyright (c) 2003-2019, CKSource - Frederico Knabben. All rights reserved.
+For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license
+*/
+CKEDITOR.plugins.setLang( 'image2', 'ko', {
+	alt: '대체 문자열',
+	btnUpload: '서버로 전송',
+	captioned: '이미지 설명 넣기',
+	captionPlaceholder: '설명',
+	infoTab: '이미지 정보',
+	lockRatio: '비율 유지',
+	menu: '이미지 속성',
+	pathName: '이미지',
+	pathNameCaption: '설명',
+	resetSize: '원래 크기로',
+	resizer: '크기를 조절하려면 클릭 후 드래그 하세요',
+	title: '이미지 속성',
+	uploadTab: '업로드',
+	urlMissing: '이미지 원본 주소(URL)가 없습니다.',
+	altMissing: '대체 문자가 없습니다.'
+} );

+ 21 - 0
public/ckeditor/plugins/image2/lang/ku.js

@@ -0,0 +1,21 @@
+/*
+Copyright (c) 2003-2019, CKSource - Frederico Knabben. All rights reserved.
+For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license
+*/
+CKEDITOR.plugins.setLang( 'image2', 'ku', {
+	alt: 'جێگرەوەی دەق',
+	btnUpload: 'ناردنی بۆ ڕاژه',
+	captioned: 'وێنەی بەسەردێر',
+	captionPlaceholder: 'سەردێر',
+	infoTab: 'زانیاری وێنه',
+	lockRatio: 'داخستنی ڕێژه',
+	menu: 'خاسیەتی وێنه',
+	pathName: 'وێنە',
+	pathNameCaption: 'سەردێر',
+	resetSize: 'ڕێکخستنەوەی قەباره',
+	resizer: 'کرتەبکە و ڕایبکێشە بۆ قەبارە گۆڕین',
+	title: 'خاسیەتی وێنه',
+	uploadTab: 'بارکردن',
+	urlMissing: 'سەرچاوەی بەستەری وێنه بزره',
+	altMissing: 'جێگرەوەی دەق لەدەست چووە.'
+} );

+ 21 - 0
public/ckeditor/plugins/image2/lang/lt.js

@@ -0,0 +1,21 @@
+/*
+Copyright (c) 2003-2019, CKSource - Frederico Knabben. All rights reserved.
+For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license
+*/
+CKEDITOR.plugins.setLang( 'image2', 'lt', {
+	alt: 'Alternatyvus Tekstas',
+	btnUpload: 'Siųsti į serverį',
+	captioned: 'Captioned image', // MISSING
+	captionPlaceholder: 'Caption', // MISSING
+	infoTab: 'Vaizdo informacija',
+	lockRatio: 'Išlaikyti proporciją',
+	menu: 'Vaizdo savybės',
+	pathName: 'image', // MISSING
+	pathNameCaption: 'caption', // MISSING
+	resetSize: 'Atstatyti dydį',
+	resizer: 'Click and drag to resize', // MISSING
+	title: 'Vaizdo savybės',
+	uploadTab: 'Siųsti',
+	urlMissing: 'Paveiksliuko nuorodos nėra.',
+	altMissing: 'Alternative text is missing.' // MISSING
+} );

+ 21 - 0
public/ckeditor/plugins/image2/lang/lv.js

@@ -0,0 +1,21 @@
+/*
+Copyright (c) 2003-2019, CKSource - Frederico Knabben. All rights reserved.
+For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license
+*/
+CKEDITOR.plugins.setLang( 'image2', 'lv', {
+	alt: 'Alternatīvais teksts',
+	btnUpload: 'Nosūtīt serverim',
+	captioned: 'Attēls ar parakstu',
+	captionPlaceholder: 'Paraksts',
+	infoTab: 'Informācija par attēlu',
+	lockRatio: 'Nemainīga Augstuma/Platuma attiecība',
+	menu: 'Attēla īpašības',
+	pathName: 'Attēls',
+	pathNameCaption: 'paraksts',
+	resetSize: 'Atjaunot sākotnējo izmēru',
+	resizer: 'Noklikšķini un pavelc lai mērogotu',
+	title: 'Attēla īpašības',
+	uploadTab: 'Augšupielādēt',
+	urlMissing: 'Trūkst attēla atrašanās adrese.',
+	altMissing: 'Trūkst alternatīvais teksts'
+} );

+ 21 - 0
public/ckeditor/plugins/image2/lang/mk.js

@@ -0,0 +1,21 @@
+/*
+Copyright (c) 2003-2019, CKSource - Frederico Knabben. All rights reserved.
+For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license
+*/
+CKEDITOR.plugins.setLang( 'image2', 'mk', {
+	alt: 'Алтернативен текст',
+	btnUpload: 'Прикачи на сервер',
+	captioned: 'Captioned image', // MISSING
+	captionPlaceholder: 'Caption', // MISSING
+	infoTab: 'Информации за сликата',
+	lockRatio: 'Зачувај пропорција',
+	menu: 'Својства на сликата',
+	pathName: 'image', // MISSING
+	pathNameCaption: 'caption', // MISSING
+	resetSize: 'Ресетирај големина',
+	resizer: 'Click and drag to resize', // MISSING
+	title: 'Својства на сликата',
+	uploadTab: 'Прикачи',
+	urlMissing: 'Недостасува URL-то на сликата.',
+	altMissing: 'Alternative text is missing.' // MISSING
+} );

+ 21 - 0
public/ckeditor/plugins/image2/lang/mn.js

@@ -0,0 +1,21 @@
+/*
+Copyright (c) 2003-2019, CKSource - Frederico Knabben. All rights reserved.
+For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license
+*/
+CKEDITOR.plugins.setLang( 'image2', 'mn', {
+	alt: 'Зургийг орлох бичвэр',
+	btnUpload: 'Үүнийг сервэррүү илгээ',
+	captioned: 'Captioned image', // MISSING
+	captionPlaceholder: 'Caption', // MISSING
+	infoTab: 'Зурагны мэдээлэл',
+	lockRatio: 'Радио түгжих',
+	menu: 'Зураг',
+	pathName: 'image', // MISSING
+	pathNameCaption: 'caption', // MISSING
+	resetSize: 'хэмжээ дахин оноох',
+	resizer: 'Click and drag to resize', // MISSING
+	title: 'Зураг',
+	uploadTab: 'Илгээж ачаалах',
+	urlMissing: 'Зургийн эх сурвалжийн хаяг (URL) байхгүй байна.',
+	altMissing: 'Alternative text is missing.' // MISSING
+} );

+ 21 - 0
public/ckeditor/plugins/image2/lang/ms.js

@@ -0,0 +1,21 @@
+/*
+Copyright (c) 2003-2019, CKSource - Frederico Knabben. All rights reserved.
+For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license
+*/
+CKEDITOR.plugins.setLang( 'image2', 'ms', {
+	alt: 'Text Alternatif',
+	btnUpload: 'Hantar ke Server',
+	captioned: 'Captioned image', // MISSING
+	captionPlaceholder: 'Caption', // MISSING
+	infoTab: 'Info Imej',
+	lockRatio: 'Tetapkan Nisbah',
+	menu: 'Ciri-ciri Imej',
+	pathName: 'image', // MISSING
+	pathNameCaption: 'caption', // MISSING
+	resetSize: 'Saiz Set Semula',
+	resizer: 'Click and drag to resize', // MISSING
+	title: 'Ciri-ciri Imej',
+	uploadTab: 'Muat Naik',
+	urlMissing: 'Image source URL is missing.', // MISSING
+	altMissing: 'Alternative text is missing.' // MISSING
+} );

+ 21 - 0
public/ckeditor/plugins/image2/lang/nb.js

@@ -0,0 +1,21 @@
+/*
+Copyright (c) 2003-2019, CKSource - Frederico Knabben. All rights reserved.
+For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license
+*/
+CKEDITOR.plugins.setLang( 'image2', 'nb', {
+	alt: 'Alternativ tekst',
+	btnUpload: 'Send det til serveren',
+	captioned: 'Bilde med bildetekst',
+	captionPlaceholder: 'Bildetekst',
+	infoTab: 'Bildeinformasjon',
+	lockRatio: 'Lås forhold',
+	menu: 'Bildeegenskaper',
+	pathName: 'bilde',
+	pathNameCaption: 'bildetekst',
+	resetSize: 'Tilbakestill størrelse',
+	resizer: 'Klikk og dra for å endre størrelse',
+	title: 'Bildeegenskaper',
+	uploadTab: 'Last opp',
+	urlMissing: 'Bildets adresse mangler.',
+	altMissing: 'Alternativ tekst mangler.'
+} );

+ 21 - 0
public/ckeditor/plugins/image2/lang/nl.js

@@ -0,0 +1,21 @@
+/*
+Copyright (c) 2003-2019, CKSource - Frederico Knabben. All rights reserved.
+For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license
+*/
+CKEDITOR.plugins.setLang( 'image2', 'nl', {
+	alt: 'Alternatieve tekst',
+	btnUpload: 'Naar server verzenden',
+	captioned: 'Afbeelding met onderschrift',
+	captionPlaceholder: 'Onderschrift',
+	infoTab: 'Afbeeldingsinformatie',
+	lockRatio: 'Verhouding vergrendelen',
+	menu: 'Eigenschappen afbeelding',
+	pathName: 'afbeelding',
+	pathNameCaption: 'onderschrift',
+	resetSize: 'Afmetingen herstellen',
+	resizer: 'Klik en sleep om te herschalen',
+	title: 'Afbeeldingseigenschappen',
+	uploadTab: 'Uploaden',
+	urlMissing: 'De URL naar de afbeelding ontbreekt.',
+	altMissing: 'Alternatieve tekst ontbreekt.'
+} );

+ 21 - 0
public/ckeditor/plugins/image2/lang/no.js

@@ -0,0 +1,21 @@
+/*
+Copyright (c) 2003-2019, CKSource - Frederico Knabben. All rights reserved.
+For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license
+*/
+CKEDITOR.plugins.setLang( 'image2', 'no', {
+	alt: 'Alternativ tekst',
+	btnUpload: 'Send det til serveren',
+	captioned: 'Bilde med bildetekst',
+	captionPlaceholder: 'Billedtekst',
+	infoTab: 'Bildeinformasjon',
+	lockRatio: 'Lås forhold',
+	menu: 'Bildeegenskaper',
+	pathName: 'bilde',
+	pathNameCaption: 'bildetekst',
+	resetSize: 'Tilbakestill størrelse',
+	resizer: 'Klikk og dra for å endre størrelse',
+	title: 'Bildeegenskaper',
+	uploadTab: 'Last opp',
+	urlMissing: 'Bildets adresse mangler.',
+	altMissing: 'Alternativ tekst mangler.'
+} );

+ 21 - 0
public/ckeditor/plugins/image2/lang/oc.js

@@ -0,0 +1,21 @@
+/*
+Copyright (c) 2003-2019, CKSource - Frederico Knabben. All rights reserved.
+For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license
+*/
+CKEDITOR.plugins.setLang( 'image2', 'oc', {
+	alt: 'Tèxte alternatiu',
+	btnUpload: 'Mandar sul servidor',
+	captioned: 'Imatge amb legenda',
+	captionPlaceholder: 'Legenda',
+	infoTab: 'Informacions sus l\'imatge',
+	lockRatio: 'Conservar las proporcions',
+	menu: 'Proprietats de l\'imatge',
+	pathName: 'imatge',
+	pathNameCaption: 'legenda',
+	resetSize: 'Reïnicializar la talha',
+	resizer: 'Clicar e lisar per redimensionar',
+	title: 'Proprietats de l\'imatge',
+	uploadTab: 'Mandar',
+	urlMissing: 'L\'URL font de l\'imatge es mancanta.',
+	altMissing: 'Lo tèxte alternatiu es mancant.'
+} );

+ 21 - 0
public/ckeditor/plugins/image2/lang/pl.js

@@ -0,0 +1,21 @@
+/*
+Copyright (c) 2003-2019, CKSource - Frederico Knabben. All rights reserved.
+For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license
+*/
+CKEDITOR.plugins.setLang( 'image2', 'pl', {
+	alt: 'Tekst zastępczy',
+	btnUpload: 'Wyślij',
+	captioned: 'Obrazek z podpisem',
+	captionPlaceholder: 'Podpis',
+	infoTab: 'Informacje o obrazku',
+	lockRatio: 'Zablokuj proporcje',
+	menu: 'Właściwości obrazka',
+	pathName: 'obrazek',
+	pathNameCaption: 'podpis',
+	resetSize: 'Przywróć rozmiar',
+	resizer: 'Kliknij i przeciągnij, by zmienić rozmiar.',
+	title: 'Właściwości obrazka',
+	uploadTab: 'Wyślij',
+	urlMissing: 'Podaj adres URL obrazka.',
+	altMissing: 'Podaj tekst zastępczy obrazka.'
+} );

+ 21 - 0
public/ckeditor/plugins/image2/lang/pt-br.js

@@ -0,0 +1,21 @@
+/*
+Copyright (c) 2003-2019, CKSource - Frederico Knabben. All rights reserved.
+For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license
+*/
+CKEDITOR.plugins.setLang( 'image2', 'pt-br', {
+	alt: 'Texto Alternativo',
+	btnUpload: 'Enviar para o Servidor',
+	captioned: 'Legenda da Imagem',
+	captionPlaceholder: 'Legenda',
+	infoTab: 'Informações da Imagem',
+	lockRatio: 'Travar Proporções',
+	menu: 'Formatar Imagem',
+	pathName: 'Imagem',
+	pathNameCaption: 'Legenda',
+	resetSize: 'Redefinir para o Tamanho Original',
+	resizer: 'Click e arraste para redimensionar',
+	title: 'Formatar Imagem',
+	uploadTab: 'Enviar ao Servidor',
+	urlMissing: 'URL da imagem está faltando.',
+	altMissing: 'Texto alternativo não informado.'
+} );

+ 21 - 0
public/ckeditor/plugins/image2/lang/pt.js

@@ -0,0 +1,21 @@
+/*
+Copyright (c) 2003-2019, CKSource - Frederico Knabben. All rights reserved.
+For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license
+*/
+CKEDITOR.plugins.setLang( 'image2', 'pt', {
+	alt: 'Texto alternativo',
+	btnUpload: 'Enviar para o servidor',
+	captioned: 'Imagem legendada',
+	captionPlaceholder: 'Legenda',
+	infoTab: 'Informação da imagem',
+	lockRatio: 'Proporcional',
+	menu: 'Propriedades da imagem',
+	pathName: 'imagem',
+	pathNameCaption: 'legenda',
+	resetSize: 'Tamanho original',
+	resizer: 'Clique e arraste para redimensionar',
+	title: 'Propriedades da imagem',
+	uploadTab: 'Carregar',
+	urlMissing: 'O URL de origem da imagem está em falta.',
+	altMissing: 'Texto alternativo em falta.'
+} );

+ 21 - 0
public/ckeditor/plugins/image2/lang/ro.js

@@ -0,0 +1,21 @@
+/*
+Copyright (c) 2003-2019, CKSource - Frederico Knabben. All rights reserved.
+For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license
+*/
+CKEDITOR.plugins.setLang( 'image2', 'ro', {
+	alt: 'Text alternativ',
+	btnUpload: 'Încarcă pe server',
+	captioned: 'Descris',
+	captionPlaceholder: 'Descriere implicită',
+	infoTab: 'Informaţii despre imagine',
+	lockRatio: 'Păstrează proporţiile',
+	menu: 'Proprietăţile imaginii',
+	pathName: 'Adresa căii',
+	pathNameCaption: 'Descrierea numelui căii',
+	resetSize: 'Resetează mărimea',
+	resizer: 'Redimensionare dinamică',
+	title: 'Proprietăţile imaginii',
+	uploadTab: 'Încarcă',
+	urlMissing: 'Sursa URL a imaginii lipsește.',
+	altMissing: 'Textul alternativ descriptive lipsește!'
+} );

+ 21 - 0
public/ckeditor/plugins/image2/lang/ru.js

@@ -0,0 +1,21 @@
+/*
+Copyright (c) 2003-2019, CKSource - Frederico Knabben. All rights reserved.
+For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license
+*/
+CKEDITOR.plugins.setLang( 'image2', 'ru', {
+	alt: 'Альтернативный текст',
+	btnUpload: 'Загрузить на сервер',
+	captioned: 'Отображать название',
+	captionPlaceholder: 'Название',
+	infoTab: 'Данные об изображении',
+	lockRatio: 'Сохранять пропорции',
+	menu: 'Свойства изображения',
+	pathName: 'изображение',
+	pathNameCaption: 'название',
+	resetSize: 'Вернуть обычные размеры',
+	resizer: 'Нажмите и растяните',
+	title: 'Свойства изображения',
+	uploadTab: 'Загрузка файла',
+	urlMissing: 'Не указана ссылка на изображение.',
+	altMissing: 'Не задан альтернативный текст'
+} );

+ 21 - 0
public/ckeditor/plugins/image2/lang/si.js

@@ -0,0 +1,21 @@
+/*
+Copyright (c) 2003-2019, CKSource - Frederico Knabben. All rights reserved.
+For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license
+*/
+CKEDITOR.plugins.setLang( 'image2', 'si', {
+	alt: 'විකල්ප ',
+	btnUpload: 'සේවාදායකය වෙත යොමුකිරිම',
+	captioned: 'Captioned image', // MISSING
+	captionPlaceholder: 'Caption', // MISSING
+	infoTab: 'රුපයේ තොරතුරු',
+	lockRatio: 'නවතන අනුපාතය ',
+	menu: 'රුපයේ ගුණ',
+	pathName: 'image', // MISSING
+	pathNameCaption: 'caption', // MISSING
+	resetSize: 'නැවතත් විශාලත්වය වෙනස් කිරීම',
+	resizer: 'Click and drag to resize', // MISSING
+	title: 'රුපයේ ',
+	uploadTab: 'උඩුගතකිරීම',
+	urlMissing: 'රුප මුලාශ්‍ර URL නැත.',
+	altMissing: 'Alternative text is missing.' // MISSING
+} );

+ 21 - 0
public/ckeditor/plugins/image2/lang/sk.js

@@ -0,0 +1,21 @@
+/*
+Copyright (c) 2003-2019, CKSource - Frederico Knabben. All rights reserved.
+For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license
+*/
+CKEDITOR.plugins.setLang( 'image2', 'sk', {
+	alt: 'Alternatívny text',
+	btnUpload: 'Odoslať to na server',
+	captioned: 'Opísaný obrázok',
+	captionPlaceholder: 'Popis',
+	infoTab: 'Informácie o obrázku',
+	lockRatio: 'Pomer zámky',
+	menu: 'Vlastnosti obrázka',
+	pathName: 'obrázok',
+	pathNameCaption: 'popis',
+	resetSize: 'Pôvodná veľkosť',
+	resizer: 'Kliknite a potiahnite pre zmenu veľkosti',
+	title: 'Vlastnosti obrázka',
+	uploadTab: 'Nahrať',
+	urlMissing: 'Chýba URL zdroja obrázka.',
+	altMissing: 'Chýba alternatívny text.'
+} );

+ 21 - 0
public/ckeditor/plugins/image2/lang/sl.js

@@ -0,0 +1,21 @@
+/*
+Copyright (c) 2003-2019, CKSource - Frederico Knabben. All rights reserved.
+For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license
+*/
+CKEDITOR.plugins.setLang( 'image2', 'sl', {
+	alt: 'Nadomestno besedilo',
+	btnUpload: 'Pošlji na strežnik',
+	captioned: 'Slika z napisom',
+	captionPlaceholder: 'Napis',
+	infoTab: 'Podatki o sliki',
+	lockRatio: 'Zakleni razmerje',
+	menu: 'Lastnosti slike',
+	pathName: 'slika',
+	pathNameCaption: 'napis',
+	resetSize: 'Ponastavi velikost',
+	resizer: 'Kliknite in povlecite, da spremenite velikost',
+	title: 'Lastnosti slike',
+	uploadTab: 'Naloži',
+	urlMissing: 'Manjka vir (URL) slike.',
+	altMissing: 'Alternative text is missing.' // MISSING
+} );

+ 21 - 0
public/ckeditor/plugins/image2/lang/sq.js

@@ -0,0 +1,21 @@
+/*
+Copyright (c) 2003-2019, CKSource - Frederico Knabben. All rights reserved.
+For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license
+*/
+CKEDITOR.plugins.setLang( 'image2', 'sq', {
+	alt: 'Tekst Alternativ',
+	btnUpload: 'Dërgo në server',
+	captioned: 'Foto e titulluar',
+	captionPlaceholder: 'Titulli',
+	infoTab: 'Informacione mbi Fotografinë',
+	lockRatio: 'Mbyll Racionin',
+	menu: 'Karakteristikat e Fotografisë',
+	pathName: 'foto',
+	pathNameCaption: 'titull',
+	resetSize: 'Rikthe Madhësinë',
+	resizer: 'Kliko dhe tërhiqe për ndryshim të madhësisë',
+	title: 'Karakteristikat e Fotografisë',
+	uploadTab: 'Ngarko',
+	urlMissing: 'Mungon URL e burimit të fotografisë.',
+	altMissing: 'Teksti alternativ mungon.'
+} );

+ 21 - 0
public/ckeditor/plugins/image2/lang/sr-latn.js

@@ -0,0 +1,21 @@
+/*
+Copyright (c) 2003-2019, CKSource - Frederico Knabben. All rights reserved.
+For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license
+*/
+CKEDITOR.plugins.setLang( 'image2', 'sr-latn', {
+	alt: 'Alternativni tekst',
+	btnUpload: 'Pošalji na server',
+	captioned: 'Slika sa natpisom',
+	captionPlaceholder: 'Natpis',
+	infoTab: 'Osnovne karakteristike',
+	lockRatio: 'Zadrži odnos',
+	menu: 'Osobine slike',
+	pathName: 'Slika',
+	pathNameCaption: 'Natpis',
+	resetSize: 'Original  veličina',
+	resizer: 'Kliknite i povucite da bi ste promenili veličinu',
+	title: 'Osobine slika',
+	uploadTab: 'Postavi',
+	urlMissing: 'Nedostaje URL slike',
+	altMissing: 'Nedostaje alternativni tekst'
+} );

+ 21 - 0
public/ckeditor/plugins/image2/lang/sr.js

@@ -0,0 +1,21 @@
+/*
+Copyright (c) 2003-2019, CKSource - Frederico Knabben. All rights reserved.
+For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license
+*/
+CKEDITOR.plugins.setLang( 'image2', 'sr', {
+	alt: 'Алтернативни текст',
+	btnUpload: 'Пошаљи на сервер',
+	captioned: 'Слика са натписом',
+	captionPlaceholder: 'Натпис',
+	infoTab: 'Основне карактеристике',
+	lockRatio: 'Задржи однос',
+	menu: 'Особине слика',
+	pathName: 'Слика',
+	pathNameCaption: 'Натпис',
+	resetSize: 'Оригинална величина',
+	resizer: 'Кликните и повуците да би сте променили величину',
+	title: 'Карактеристике слике',
+	uploadTab: 'Постави',
+	urlMissing: 'Недостаје УРЛ слике.',
+	altMissing: 'Недостаје алтернативни текст.'
+} );

+ 21 - 0
public/ckeditor/plugins/image2/lang/sv.js

@@ -0,0 +1,21 @@
+/*
+Copyright (c) 2003-2019, CKSource - Frederico Knabben. All rights reserved.
+For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license
+*/
+CKEDITOR.plugins.setLang( 'image2', 'sv', {
+	alt: 'Alternativ text',
+	btnUpload: 'Skicka till server',
+	captioned: 'Rubricerad bild',
+	captionPlaceholder: 'Bildtext',
+	infoTab: 'Bildinformation',
+	lockRatio: 'Lås höjd/bredd förhållanden',
+	menu: 'Bildegenskaper',
+	pathName: 'bild',
+	pathNameCaption: 'rubrik',
+	resetSize: 'Återställ storlek',
+	resizer: 'Klicka och drag för att ändra storlek',
+	title: 'Bildegenskaper',
+	uploadTab: 'Ladda upp',
+	urlMissing: 'Bildkällans URL saknas.',
+	altMissing: 'Alternativ text saknas'
+} );

+ 21 - 0
public/ckeditor/plugins/image2/lang/th.js

@@ -0,0 +1,21 @@
+/*
+Copyright (c) 2003-2019, CKSource - Frederico Knabben. All rights reserved.
+For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license
+*/
+CKEDITOR.plugins.setLang( 'image2', 'th', {
+	alt: 'คำประกอบรูปภาพ',
+	btnUpload: 'อัพโหลดไฟล์ไปเก็บไว้ที่เครื่องแม่ข่าย (เซิร์ฟเวอร์)',
+	captioned: 'Captioned image', // MISSING
+	captionPlaceholder: 'Caption', // MISSING
+	infoTab: 'ข้อมูลของรูปภาพ',
+	lockRatio: 'กำหนดอัตราส่วน กว้าง-สูง แบบคงที่',
+	menu: 'คุณสมบัติของ รูปภาพ',
+	pathName: 'image', // MISSING
+	pathNameCaption: 'caption', // MISSING
+	resetSize: 'กำหนดรูปเท่าขนาดจริง',
+	resizer: 'Click and drag to resize', // MISSING
+	title: 'คุณสมบัติของ รูปภาพ',
+	uploadTab: 'อัพโหลดไฟล์',
+	urlMissing: 'Image source URL is missing.', // MISSING
+	altMissing: 'Alternative text is missing.' // MISSING
+} );

+ 21 - 0
public/ckeditor/plugins/image2/lang/tr.js

@@ -0,0 +1,21 @@
+/*
+Copyright (c) 2003-2019, CKSource - Frederico Knabben. All rights reserved.
+For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license
+*/
+CKEDITOR.plugins.setLang( 'image2', 'tr', {
+	alt: 'Alternatif Yazı',
+	btnUpload: 'Sunucuya Yolla',
+	captioned: 'Başlıklı resim',
+	captionPlaceholder: 'Başlık',
+	infoTab: 'Resim Bilgisi',
+	lockRatio: 'Oranı Kilitle',
+	menu: 'Resim Özellikleri',
+	pathName: 'Resim',
+	pathNameCaption: 'başlık',
+	resetSize: 'Boyutu Başa Döndür',
+	resizer: 'Boyutlandırmak için, tıklayın ve sürükleyin',
+	title: 'Resim Özellikleri',
+	uploadTab: 'Karşıya Yükle',
+	urlMissing: 'Resmin URL kaynağı bulunamadı.',
+	altMissing: 'Alternatif yazı eksik.'
+} );

+ 21 - 0
public/ckeditor/plugins/image2/lang/tt.js

@@ -0,0 +1,21 @@
+/*
+Copyright (c) 2003-2019, CKSource - Frederico Knabben. All rights reserved.
+For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license
+*/
+CKEDITOR.plugins.setLang( 'image2', 'tt', {
+	alt: 'Альтернатив текст',
+	btnUpload: 'Серверга җибәрү',
+	captioned: 'Исеме куелган рәсем',
+	captionPlaceholder: 'Исем',
+	infoTab: 'Рәсем тасвирламасы',
+	lockRatio: 'Lock Ratio', // MISSING
+	menu: 'Рәсем үзлекләре',
+	pathName: 'рәсем',
+	pathNameCaption: 'исем',
+	resetSize: 'Баштагы зурлык',
+	resizer: 'Күчереп куер өчен басып шудырыгыз',
+	title: 'Рәсем үзлекләре',
+	uploadTab: 'Йөкләү',
+	urlMissing: 'Image source URL is missing.', // MISSING
+	altMissing: 'Alternative text is missing.' // MISSING
+} );

+ 21 - 0
public/ckeditor/plugins/image2/lang/ug.js

@@ -0,0 +1,21 @@
+/*
+Copyright (c) 2003-2019, CKSource - Frederico Knabben. All rights reserved.
+For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license
+*/
+CKEDITOR.plugins.setLang( 'image2', 'ug', {
+	alt: 'تېكىست ئالماشتۇر',
+	btnUpload: 'مۇلازىمېتىرغا يۈكلە',
+	captioned: 'ماۋزۇلۇق سۈرەت',
+	captionPlaceholder: 'ماۋزۇ',
+	infoTab: 'سۈرەت',
+	lockRatio: 'نىسبەتنى قۇلۇپلا',
+	menu: 'سۈرەت خاسلىقى',
+	pathName: 'رەسىم',
+	pathNameCaption: 'ماۋزۇ',
+	resetSize: 'ئەسلى چوڭلۇق',
+	resizer: 'چېكىپ تۇرۇپ سۆرەپ چوڭ كىچىكلىكىنى تەڭشىگىلى بولىدۇ',
+	title: 'سۈرەت خاسلىقى',
+	uploadTab: 'يۈكلە',
+	urlMissing: 'سۈرەتنىڭ ئەسلى ھۆججەت ئادرېسى كەم',
+	altMissing: 'باشقا تېكىست كەمچىل'
+} );

+ 21 - 0
public/ckeditor/plugins/image2/lang/uk.js

@@ -0,0 +1,21 @@
+/*
+Copyright (c) 2003-2019, CKSource - Frederico Knabben. All rights reserved.
+For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license
+*/
+CKEDITOR.plugins.setLang( 'image2', 'uk', {
+	alt: 'Альтернативний текст',
+	btnUpload: 'Надіслати на сервер',
+	captioned: 'Підписане зображення',
+	captionPlaceholder: 'Заголовок',
+	infoTab: 'Інформація про зображення',
+	lockRatio: 'Зберегти пропорції',
+	menu: 'Властивості зображення',
+	pathName: 'Зображення',
+	pathNameCaption: 'заголовок',
+	resetSize: 'Очистити поля розмірів',
+	resizer: 'Клікніть та потягніть для зміни розмірів',
+	title: 'Властивості зображення',
+	uploadTab: 'Надіслати',
+	urlMissing: 'Вкажіть URL зображення.',
+	altMissing: 'Альтернативний текст відсутній.'
+} );

+ 21 - 0
public/ckeditor/plugins/image2/lang/vi.js

@@ -0,0 +1,21 @@
+/*
+Copyright (c) 2003-2019, CKSource - Frederico Knabben. All rights reserved.
+For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license
+*/
+CKEDITOR.plugins.setLang( 'image2', 'vi', {
+	alt: 'Chú thích ảnh',
+	btnUpload: 'Tải lên máy chủ',
+	captioned: 'Ảnh có chú thích',
+	captionPlaceholder: 'Nhãn',
+	infoTab: 'Thông tin của ảnh',
+	lockRatio: 'Giữ nguyên tỷ lệ',
+	menu: 'Thuộc tính của ảnh',
+	pathName: 'ảnh',
+	pathNameCaption: 'chú thích',
+	resetSize: 'Kích thước gốc',
+	resizer: 'Kéo rê để thay đổi kích cỡ',
+	title: 'Thuộc tính của ảnh',
+	uploadTab: 'Tải lên',
+	urlMissing: 'Thiếu đường dẫn hình ảnh',
+	altMissing: 'Alternative text is missing.' // MISSING
+} );

+ 21 - 0
public/ckeditor/plugins/image2/lang/zh-cn.js

@@ -0,0 +1,21 @@
+/*
+Copyright (c) 2003-2019, CKSource - Frederico Knabben. All rights reserved.
+For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license
+*/
+CKEDITOR.plugins.setLang( 'image2', 'zh-cn', {
+	alt: '替换文本',
+	btnUpload: '上传到服务器',
+	captioned: '带标题图像',
+	captionPlaceholder: '标题',
+	infoTab: '图像信息',
+	lockRatio: '锁定比例',
+	menu: '图像属性',
+	pathName: '图像',
+	pathNameCaption: '标题',
+	resetSize: '原始尺寸',
+	resizer: '点击并拖拽以改变尺寸',
+	title: '图像属性',
+	uploadTab: '上传',
+	urlMissing: '缺少图像源文件地址',
+	altMissing: '缺少替换文本'
+} );

+ 21 - 0
public/ckeditor/plugins/image2/lang/zh.js

@@ -0,0 +1,21 @@
+/*
+Copyright (c) 2003-2019, CKSource - Frederico Knabben. All rights reserved.
+For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license
+*/
+CKEDITOR.plugins.setLang( 'image2', 'zh', {
+	alt: '替代文字',
+	btnUpload: '傳送至伺服器',
+	captioned: '已加標題之圖片',
+	captionPlaceholder: '標題',
+	infoTab: '影像資訊',
+	lockRatio: '固定比例',
+	menu: '影像屬性',
+	pathName: '圖片',
+	pathNameCaption: '標題',
+	resetSize: '重設大小',
+	resizer: '拖曳以改變大小',
+	title: '影像屬性',
+	uploadTab: '上傳',
+	urlMissing: '遺失圖片來源之 URL ',
+	altMissing: '替代文字遺失。'
+} );

+ 1783 - 0
public/ckeditor/plugins/image2/plugin.js

@@ -0,0 +1,1783 @@
+/**
+ * @license Copyright (c) 2003-2019, CKSource - Frederico Knabben. All rights reserved.
+ * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license
+ */
+
+'use strict';
+
+( function() {
+
+	var template = '<img alt="" src="" />',
+		templateBlock = new CKEDITOR.template(
+			'<figure class="{captionedClass}">' +
+				template +
+				'<figcaption>{captionPlaceholder}</figcaption>' +
+			'</figure>' ),
+		alignmentsObj = { left: 0, center: 1, right: 2 },
+		regexPercent = /^\s*(\d+\%)\s*$/i;
+
+	CKEDITOR.plugins.add( 'image2', {
+		// jscs:disable maximumLineLength
+		lang: 'af,ar,az,bg,bn,bs,ca,cs,cy,da,de,de-ch,el,en,en-au,en-ca,en-gb,eo,es,es-mx,et,eu,fa,fi,fo,fr,fr-ca,gl,gu,he,hi,hr,hu,id,is,it,ja,ka,km,ko,ku,lt,lv,mk,mn,ms,nb,nl,no,oc,pl,pt,pt-br,ro,ru,si,sk,sl,sq,sr,sr-latn,sv,th,tr,tt,ug,uk,vi,zh,zh-cn', // %REMOVE_LINE_CORE%
+		// jscs:enable maximumLineLength
+		requires: 'widget,dialog',
+		icons: 'image',
+		hidpi: true,
+
+		onLoad: function() {
+			CKEDITOR.addCss(
+			'.cke_image_nocaption{' +
+				// This is to remove unwanted space so resize
+				// wrapper is displayed property.
+				'line-height:0' +
+			'}' +
+			'.cke_editable.cke_image_sw, .cke_editable.cke_image_sw *{cursor:sw-resize !important}' +
+			'.cke_editable.cke_image_se, .cke_editable.cke_image_se *{cursor:se-resize !important}' +
+			'.cke_image_resizer{' +
+				'display:none;' +
+				'position:absolute;' +
+				'width:10px;' +
+				'height:10px;' +
+				'bottom:-5px;' +
+				'right:-5px;' +
+				'background:#000;' +
+				'outline:1px solid #fff;' +
+				// Prevent drag handler from being misplaced (https://dev.ckeditor.com/ticket/11207).
+				'line-height:0;' +
+				'cursor:se-resize;' +
+			'}' +
+			'.cke_image_resizer_wrapper{' +
+				'position:relative;' +
+				'display:inline-block;' +
+				'line-height:0;' +
+			'}' +
+			// Bottom-left corner style of the resizer.
+			'.cke_image_resizer.cke_image_resizer_left{' +
+				'right:auto;' +
+				'left:-5px;' +
+				'cursor:sw-resize;' +
+			'}' +
+			'.cke_widget_wrapper:hover .cke_image_resizer,' +
+			'.cke_image_resizer.cke_image_resizing{' +
+				'display:block' +
+			'}' +
+			// Hide resizer in read only mode (#2816).
+			'.cke_editable[contenteditable="false"] .cke_image_resizer{' +
+				'display:none;' +
+			'}' +
+			// Expand widget wrapper when linked inline image.
+			'.cke_widget_wrapper>a{' +
+				'display:inline-block' +
+			'}' );
+		},
+
+		init: function( editor ) {
+			// Abort when Easyimage is to be loaded since this plugins
+			// share the same functionality (#1791).
+			if ( editor.plugins.detectConflict( 'image2', [ 'easyimage' ] ) ) {
+				return;
+			}
+
+			// Adapts configuration from original image plugin. Should be removed
+			// when we'll rename image2 to image.
+			var config = editor.config,
+				lang = editor.lang.image2,
+				image = widgetDef( editor );
+
+			// Since filebrowser plugin discovers config properties by dialog (plugin?)
+			// names (sic!), this hack will be necessary as long as Image2 is not named
+			// Image. And since Image2 will never be Image, for sure some filebrowser logic
+			// got to be refined.
+			config.filebrowserImage2BrowseUrl = config.filebrowserImageBrowseUrl;
+			config.filebrowserImage2UploadUrl = config.filebrowserImageUploadUrl;
+
+			// Add custom elementspath names to widget definition.
+			image.pathName = lang.pathName;
+			image.editables.caption.pathName = lang.pathNameCaption;
+
+			// Register the widget.
+			editor.widgets.add( 'image', image );
+
+			// Add toolbar button for this plugin.
+			editor.ui.addButton && editor.ui.addButton( 'Image', {
+				label: editor.lang.common.image,
+				command: 'image',
+				toolbar: 'insert,10'
+			} );
+
+			// Register context menu option for editing widget.
+			if ( editor.contextMenu ) {
+				editor.addMenuGroup( 'image', 10 );
+
+				editor.addMenuItem( 'image', {
+					label: lang.menu,
+					command: 'image',
+					group: 'image'
+				} );
+			}
+
+			CKEDITOR.dialog.add( 'image2', this.path + 'dialogs/image2.js' );
+		},
+
+		afterInit: function( editor ) {
+			// Integrate with align commands (justify plugin).
+			var align = { left: 1, right: 1, center: 1, block: 1 },
+				integrate = alignCommandIntegrator( editor );
+
+			for ( var value in align )
+				integrate( value );
+
+			// Integrate with link commands (link plugin).
+			linkCommandIntegrator( editor );
+		}
+	} );
+
+	// Wiget states (forms) depending on alignment and configuration.
+	//
+	// Non-captioned widget (inline styles)
+	// 		┌──────┬───────────────────────────────┬─────────────────────────────┐
+	// 		│Align │Internal form                  │Data                         │
+	// 		├──────┼───────────────────────────────┼─────────────────────────────┤
+	// 		│none  │<wrapper>                      │<img />                      │
+	// 		│      │ <img />                       │                             │
+	// 		│      │</wrapper>                     │                             │
+	// 		├──────┼───────────────────────────────┼─────────────────────────────┤
+	// 		│left  │<wrapper style=”float:left”>   │<img style=”float:left” />   │
+	// 		│      │ <img />                       │                             │
+	// 		│      │</wrapper>                     │                             │
+	// 		├──────┼───────────────────────────────┼─────────────────────────────┤
+	// 		│center│<wrapper>                      │<p style=”text-align:center”>│
+	// 		│      │ <p style=”text-align:center”> │  <img />                    │
+	// 		│      │   <img />                     │</p>                         │
+	// 		│      │ </p>                          │                             │
+	// 		│      │</wrapper>                     │                             │
+	// 		├──────┼───────────────────────────────┼─────────────────────────────┤
+	// 		│right │<wrapper style=”float:right”>  │<img style=”float:right” />  │
+	// 		│      │ <img />                       │                             │
+	// 		│      │</wrapper>                     │                             │
+	// 		└──────┴───────────────────────────────┴─────────────────────────────┘
+	//
+	// Non-captioned widget (config.image2_alignClasses defined)
+	// 		┌──────┬───────────────────────────────┬─────────────────────────────┐
+	// 		│Align │Internal form                  │Data                         │
+	// 		├──────┼───────────────────────────────┼─────────────────────────────┤
+	// 		│none  │<wrapper>                      │<img />                      │
+	// 		│      │ <img />                       │                             │
+	// 		│      │</wrapper>                     │                             │
+	// 		├──────┼───────────────────────────────┼─────────────────────────────┤
+	// 		│left  │<wrapper class=”left”>         │<img class=”left” />         │
+	// 		│      │ <img />                       │                             │
+	// 		│      │</wrapper>                     │                             │
+	// 		├──────┼───────────────────────────────┼─────────────────────────────┤
+	// 		│center│<wrapper>                      │<p class=”center”>           │
+	// 		│      │ <p class=”center”>            │ <img />                     │
+	// 		│      │   <img />                     │</p>                         │
+	// 		│      │ </p>                          │                             │
+	// 		│      │</wrapper>                     │                             │
+	// 		├──────┼───────────────────────────────┼─────────────────────────────┤
+	// 		│right │<wrapper class=”right”>        │<img class=”right” />        │
+	// 		│      │ <img />                       │                             │
+	// 		│      │</wrapper>                     │                             │
+	// 		└──────┴───────────────────────────────┴─────────────────────────────┘
+	//
+	// Captioned widget (inline styles)
+	// 		┌──────┬────────────────────────────────────────┬────────────────────────────────────────┐
+	// 		│Align │Internal form                           │Data                                    │
+	// 		├──────┼────────────────────────────────────────┼────────────────────────────────────────┤
+	// 		│none  │<wrapper>                               │<figure />                              │
+	// 		│      │ <figure />                             │                                        │
+	// 		│      │</wrapper>                              │                                        │
+	// 		├──────┼────────────────────────────────────────┼────────────────────────────────────────┤
+	// 		│left  │<wrapper style=”float:left”>            │<figure style=”float:left” />           │
+	// 		│      │ <figure />                             │                                        │
+	// 		│      │</wrapper>                              │                                        │
+	// 		├──────┼────────────────────────────────────────┼────────────────────────────────────────┤
+	// 		│center│<wrapper style=”text-align:center”>     │<div style=”text-align:center”>         │
+	// 		│      │ <figure style=”display:inline-block” />│ <figure style=”display:inline-block” />│
+	// 		│      │</wrapper>                              │</p>                                    │
+	// 		├──────┼────────────────────────────────────────┼────────────────────────────────────────┤
+	// 		│right │<wrapper style=”float:right”>           │<figure style=”float:right” />          │
+	// 		│      │ <figure />                             │                                        │
+	// 		│      │</wrapper>                              │                                        │
+	// 		└──────┴────────────────────────────────────────┴────────────────────────────────────────┘
+	//
+	// Captioned widget (config.image2_alignClasses defined)
+	// 		┌──────┬────────────────────────────────────────┬────────────────────────────────────────┐
+	// 		│Align │Internal form                           │Data                                    │
+	// 		├──────┼────────────────────────────────────────┼────────────────────────────────────────┤
+	// 		│none  │<wrapper>                               │<figure />                              │
+	// 		│      │ <figure />                             │                                        │
+	// 		│      │</wrapper>                              │                                        │
+	// 		├──────┼────────────────────────────────────────┼────────────────────────────────────────┤
+	// 		│left  │<wrapper class=”left”>                  │<figure class=”left” />                 │
+	// 		│      │ <figure />                             │                                        │
+	// 		│      │</wrapper>                              │                                        │
+	// 		├──────┼────────────────────────────────────────┼────────────────────────────────────────┤
+	// 		│center│<wrapper class=”center”>                │<div class=”center”>                    │
+	// 		│      │ <figure />                             │ <figure />                             │
+	// 		│      │</wrapper>                              │</p>                                    │
+	// 		├──────┼────────────────────────────────────────┼────────────────────────────────────────┤
+	// 		│right │<wrapper class=”right”>                 │<figure class=”right” />                │
+	// 		│      │ <figure />                             │                                        │
+	// 		│      │</wrapper>                              │                                        │
+	// 		└──────┴────────────────────────────────────────┴────────────────────────────────────────┘
+	//
+	// @param {CKEDITOR.editor}
+	// @returns {Object}
+	function widgetDef( editor ) {
+		var alignClasses = editor.config.image2_alignClasses,
+			captionedClass = editor.config.image2_captionedClass;
+
+		function deflate() {
+			if ( this.deflated )
+				return;
+
+			// Remember whether widget was focused before destroyed.
+			if ( editor.widgets.focused == this.widget )
+				this.focused = true;
+
+			editor.widgets.destroy( this.widget );
+
+			// Mark widget was destroyed.
+			this.deflated = true;
+		}
+
+		function inflate() {
+			var editable = editor.editable(),
+			doc = editor.document;
+
+			// Create a new widget. This widget will be either captioned
+			// non-captioned, block or inline according to what is the
+			// new state of the widget.
+			if ( this.deflated ) {
+				this.widget = editor.widgets.initOn( this.element, 'image', this.widget.data );
+
+				// Once widget was re-created, it may become an inline element without
+				// block wrapper (i.e. when unaligned, end not captioned). Let's do some
+				// sort of autoparagraphing here (https://dev.ckeditor.com/ticket/10853).
+				if ( this.widget.inline && !( new CKEDITOR.dom.elementPath( this.widget.wrapper, editable ).block ) ) {
+					var block = doc.createElement( editor.activeEnterMode == CKEDITOR.ENTER_P ? 'p' : 'div' );
+					block.replace( this.widget.wrapper );
+					this.widget.wrapper.move( block );
+				}
+
+				// The focus must be transferred from the old one (destroyed)
+				// to the new one (just created).
+				if ( this.focused ) {
+					this.widget.focus();
+					delete this.focused;
+				}
+
+				delete this.deflated;
+			}
+
+			// If now widget was destroyed just update wrapper's alignment.
+			// According to the new state.
+			else {
+				setWrapperAlign( this.widget, alignClasses );
+			}
+		}
+
+		return {
+			allowedContent: getWidgetAllowedContent( editor ),
+
+			requiredContent: 'img[src,alt]',
+
+			features: getWidgetFeatures( editor ),
+
+			styleableElements: 'img figure',
+
+			// This widget converts style-driven dimensions to attributes.
+			contentTransformations: [
+				[ 'img[width]: sizeToAttribute' ]
+			],
+
+			// This widget has an editable caption.
+			editables: {
+				caption: {
+					selector: 'figcaption',
+					allowedContent: 'br em strong sub sup u s; a[!href,target]'
+				}
+			},
+
+			parts: {
+				image: 'img',
+				caption: 'figcaption'
+				// parts#link defined in widget#init
+			},
+
+			// The name of this widget's dialog.
+			dialog: 'image2',
+
+			// Template of the widget: plain image.
+			template: template,
+
+			data: function() {
+				var features = this.features;
+
+				// Image can't be captioned when figcaption is disallowed (https://dev.ckeditor.com/ticket/11004).
+				if ( this.data.hasCaption && !editor.filter.checkFeature( features.caption ) )
+					this.data.hasCaption = false;
+
+				// Image can't be aligned when floating is disallowed (https://dev.ckeditor.com/ticket/11004).
+				if ( this.data.align != 'none' && !editor.filter.checkFeature( features.align ) )
+					this.data.align = 'none';
+
+				// Convert the internal form of the widget from the old state to the new one.
+				this.shiftState( {
+					widget: this,
+					element: this.element,
+					oldData: this.oldData,
+					newData: this.data,
+					deflate: deflate,
+					inflate: inflate
+				} );
+
+				// Update widget.parts.link since it will not auto-update unless widget
+				// is destroyed and re-inited.
+				if ( !this.data.link ) {
+					if ( this.parts.link )
+						delete this.parts.link;
+				} else {
+					if ( !this.parts.link )
+						this.parts.link = this.parts.image.getParent();
+				}
+
+				this.parts.image.setAttributes( {
+					src: this.data.src,
+
+					// This internal is required by the editor.
+					'data-cke-saved-src': this.data.src,
+
+					alt: this.data.alt
+				} );
+
+				// If shifting non-captioned -> captioned, remove classes
+				// related to styles from <img/>.
+				if ( this.oldData && !this.oldData.hasCaption && this.data.hasCaption ) {
+					for ( var c in this.data.classes )
+						this.parts.image.removeClass( c );
+				}
+
+				// Set dimensions of the image according to gathered data.
+				// Do it only when the attributes are allowed (https://dev.ckeditor.com/ticket/11004).
+				if ( editor.filter.checkFeature( features.dimension ) )
+					setDimensions( this );
+
+				// Cache current data.
+				this.oldData = CKEDITOR.tools.extend( {}, this.data );
+			},
+
+			init: function() {
+				var helpers = CKEDITOR.plugins.image2,
+					image = this.parts.image,
+					data = {
+						hasCaption: !!this.parts.caption,
+						src: image.getAttribute( 'src' ),
+						alt: image.getAttribute( 'alt' ) || '',
+						width: image.getAttribute( 'width' ) || '',
+						height: image.getAttribute( 'height' ) || '',
+
+						// Lock ratio is on by default (https://dev.ckeditor.com/ticket/10833).
+						lock: this.ready ? helpers.checkHasNaturalRatio( image ) : true
+					};
+
+				// If we used 'a' in widget#parts definition, it could happen that
+				// selected element is a child of widget.parts#caption. Since there's no clever
+				// way to solve it with CSS selectors, it's done like that. (https://dev.ckeditor.com/ticket/11783).
+				var link = image.getAscendant( 'a' );
+
+				if ( link && this.wrapper.contains( link ) )
+					this.parts.link = link;
+
+				// Depending on configuration, read style/class from element and
+				// then remove it. Removed style/class will be set on wrapper in #data listener.
+				// Note: Center alignment is detected during upcast, so only left/right cases
+				// are checked below.
+				if ( !data.align ) {
+					var alignElement = data.hasCaption ? this.element : image;
+
+					// Read the initial left/right alignment from the class set on element.
+					if ( alignClasses ) {
+						if ( alignElement.hasClass( alignClasses[ 0 ] ) ) {
+							data.align = 'left';
+						} else if ( alignElement.hasClass( alignClasses[ 2 ] ) ) {
+							data.align = 'right';
+						}
+
+						if ( data.align ) {
+							alignElement.removeClass( alignClasses[ alignmentsObj[ data.align ] ] );
+						} else {
+							data.align = 'none';
+						}
+					}
+					// Read initial float style from figure/image and then remove it.
+					else {
+						data.align = alignElement.getStyle( 'float' ) || 'none';
+						alignElement.removeStyle( 'float' );
+					}
+				}
+
+				// Update data.link object with attributes if the link has been discovered.
+				if ( editor.plugins.link && this.parts.link ) {
+					data.link = helpers.getLinkAttributesParser()( editor, this.parts.link );
+
+					// Get rid of cke_widget_* classes in data. Otherwise
+					// they might appear in link dialog.
+					var advanced = data.link.advanced;
+					if ( advanced && advanced.advCSSClasses ) {
+						advanced.advCSSClasses = CKEDITOR.tools.trim( advanced.advCSSClasses.replace( /cke_\S+/, '' ) );
+					}
+				}
+
+				// Get rid of extra vertical space when there's no caption.
+				// It will improve the look of the resizer.
+				this.wrapper[ ( data.hasCaption ? 'remove' : 'add' ) + 'Class' ]( 'cke_image_nocaption' );
+
+				this.setData( data );
+
+				// Setup dynamic image resizing with mouse.
+				// Don't initialize resizer when dimensions are disallowed (https://dev.ckeditor.com/ticket/11004).
+				if ( editor.filter.checkFeature( this.features.dimension ) && editor.config.image2_disableResizer !== true ) {
+					setupResizer( this );
+				}
+
+				this.shiftState = helpers.stateShifter( this.editor );
+
+				// Add widget editing option to its context menu.
+				this.on( 'contextMenu', function( evt ) {
+					evt.data.image = CKEDITOR.TRISTATE_OFF;
+
+					// Integrate context menu items for link.
+					// Note that widget may be wrapped in a link, which
+					// does not belong to that widget (https://dev.ckeditor.com/ticket/11814).
+					if ( this.parts.link || this.wrapper.getAscendant( 'a' ) )
+						evt.data.link = evt.data.unlink = CKEDITOR.TRISTATE_OFF;
+				} );
+			},
+
+			// Overrides default method to handle internal mutability of Image2.
+			// @see CKEDITOR.plugins.widget#addClass
+			addClass: function( className ) {
+				getStyleableElement( this ).addClass( className );
+			},
+
+			// Overrides default method to handle internal mutability of Image2.
+			// @see CKEDITOR.plugins.widget#hasClass
+			hasClass: function( className ) {
+				return getStyleableElement( this ).hasClass( className );
+			},
+
+			// Overrides default method to handle internal mutability of Image2.
+			// @see CKEDITOR.plugins.widget#removeClass
+			removeClass: function( className ) {
+				getStyleableElement( this ).removeClass( className );
+			},
+
+			// Overrides default method to handle internal mutability of Image2.
+			// @see CKEDITOR.plugins.widget#getClasses
+			getClasses: ( function() {
+				var classRegex = new RegExp( '^(' + [].concat( captionedClass, alignClasses ).join( '|' ) + ')$' );
+
+				return function() {
+					var classes = this.repository.parseElementClasses( getStyleableElement( this ).getAttribute( 'class' ) );
+
+					// Neither config.image2_captionedClass nor config.image2_alignClasses
+					// do not belong to style classes.
+					for ( var c in classes ) {
+						if ( classRegex.test( c ) )
+							delete classes[ c ];
+					}
+
+					return classes;
+				};
+			} )(),
+
+			upcast: upcastWidgetElement( editor ),
+			downcast: downcastWidgetElement( editor ),
+
+			getLabel: function() {
+				var label = ( this.data.alt || '' ) + ' ' + this.pathName;
+
+				return this.editor.lang.widget.label.replace( /%1/, label );
+			}
+		};
+	}
+
+	/**
+	 * A set of Enhanced Image (image2) plugin helpers.
+	 *
+	 * @class
+	 * @singleton
+	 */
+	CKEDITOR.plugins.image2 = {
+		stateShifter: function( editor ) {
+			// Tag name used for centering non-captioned widgets.
+			var doc = editor.document,
+				alignClasses = editor.config.image2_alignClasses,
+				captionedClass = editor.config.image2_captionedClass,
+				editable = editor.editable(),
+
+				// The order that stateActions get executed. It matters!
+				shiftables = [ 'hasCaption', 'align', 'link' ];
+
+			// Atomic procedures, one per state variable.
+			var stateActions = {
+				align: function( shift, oldValue, newValue ) {
+					var el = shift.element;
+
+					// Alignment changed.
+					if ( shift.changed.align ) {
+						// No caption in the new state.
+						if ( !shift.newData.hasCaption ) {
+							// Changed to "center" (non-captioned).
+							if ( newValue == 'center' ) {
+								shift.deflate();
+								shift.element = wrapInCentering( editor, el );
+							}
+
+							// Changed to "non-center" from "center" while caption removed.
+							if ( !shift.changed.hasCaption && oldValue == 'center' && newValue != 'center' ) {
+								shift.deflate();
+								shift.element = unwrapFromCentering( el );
+							}
+						}
+					}
+
+					// Alignment remains and "center" removed caption.
+					else if ( newValue == 'center' && shift.changed.hasCaption && !shift.newData.hasCaption ) {
+						shift.deflate();
+						shift.element = wrapInCentering( editor, el );
+					}
+
+					// Finally set display for figure.
+					if ( !alignClasses && el.is( 'figure' ) ) {
+						if ( newValue == 'center' )
+							el.setStyle( 'display', 'inline-block' );
+						else
+							el.removeStyle( 'display' );
+					}
+				},
+
+				hasCaption:	function( shift, oldValue, newValue ) {
+					// This action is for real state change only.
+					if ( !shift.changed.hasCaption )
+						return;
+
+					// Get <img/> or <a><img/></a> from widget. Note that widget element might itself
+					// be what we're looking for. Also element can be <p style="text-align:center"><a>...</a></p>.
+					var imageOrLink;
+					if ( shift.element.is( { img: 1, a: 1 } ) )
+						imageOrLink = shift.element;
+					else
+						imageOrLink =  shift.element.findOne( 'a,img' );
+
+					// Switching hasCaption always destroys the widget.
+					shift.deflate();
+
+					// There was no caption, but the caption is to be added.
+					if ( newValue ) {
+						// Create new <figure> from widget template.
+						var figure = CKEDITOR.dom.element.createFromHtml( templateBlock.output( {
+							captionedClass: captionedClass,
+							captionPlaceholder: editor.lang.image2.captionPlaceholder
+						} ), doc );
+
+						// Replace element with <figure>.
+						replaceSafely( figure, shift.element );
+
+						// Use old <img/> or <a><img/></a> instead of the one from the template,
+						// so we won't lose additional attributes.
+						imageOrLink.replace( figure.findOne( 'img' ) );
+
+						// Update widget's element.
+						shift.element = figure;
+					}
+
+					// The caption was present, but now it's to be removed.
+					else {
+						// Unwrap <img/> or <a><img/></a> from figure.
+						imageOrLink.replace( shift.element );
+
+						// Update widget's element.
+						shift.element = imageOrLink;
+					}
+				},
+
+				link: function( shift, oldValue, newValue ) {
+					if ( shift.changed.link ) {
+						var img = shift.element.is( 'img' ) ?
+								shift.element : shift.element.findOne( 'img' ),
+							link = shift.element.is( 'a' ) ?
+								shift.element : shift.element.findOne( 'a' ),
+							// Why deflate:
+							// If element is <img/>, it will be wrapped into <a>,
+							// which becomes a new widget.element.
+							// If element is <a><img/></a>, it will be unlinked
+							// so <img/> becomes a new widget.element.
+							needsDeflate = ( shift.element.is( 'a' ) && !newValue ) || ( shift.element.is( 'img' ) && newValue ),
+							newEl;
+
+						if ( needsDeflate )
+							shift.deflate();
+
+						// If unlinked the image, returned element is <img>.
+						if ( !newValue )
+							newEl = unwrapFromLink( link );
+						else {
+							// If linked the image, returned element is <a>.
+							if ( !oldValue )
+								newEl = wrapInLink( img, shift.newData.link );
+
+							// Set and remove all attributes associated with this state.
+							var attributes = CKEDITOR.plugins.image2.getLinkAttributesGetter()( editor, newValue );
+
+							if ( !CKEDITOR.tools.isEmpty( attributes.set ) )
+								( newEl || link ).setAttributes( attributes.set );
+
+							if ( attributes.removed.length )
+								( newEl || link ).removeAttributes( attributes.removed );
+						}
+
+						if ( needsDeflate )
+							shift.element = newEl;
+					}
+				}
+			};
+
+			function wrapInCentering( editor, element ) {
+				var attribsAndStyles = {};
+
+				if ( alignClasses )
+					attribsAndStyles.attributes = { 'class': alignClasses[ 1 ] };
+				else
+					attribsAndStyles.styles = { 'text-align': 'center' };
+
+				// There's no gentle way to center inline element with CSS, so create p/div
+				// that wraps widget contents and does the trick either with style or class.
+				var center = doc.createElement(
+					editor.activeEnterMode == CKEDITOR.ENTER_P ? 'p' : 'div', attribsAndStyles );
+
+				// Replace element with centering wrapper.
+				replaceSafely( center, element );
+				element.move( center );
+
+				return center;
+			}
+
+			function unwrapFromCentering( element ) {
+				var imageOrLink = element.findOne( 'a,img' );
+
+				imageOrLink.replace( element );
+
+				return imageOrLink;
+			}
+
+			// Wraps <img/> -> <a><img/></a>.
+			// Returns reference to <a>.
+			//
+			// @param {CKEDITOR.dom.element} img
+			// @param {Object} linkData
+			// @returns {CKEDITOR.dom.element}
+			function wrapInLink( img, linkData ) {
+				var link = doc.createElement( 'a', {
+					attributes: {
+						href: linkData.url
+					}
+				} );
+
+				link.replace( img );
+				img.move( link );
+
+				return link;
+			}
+
+			// De-wraps <a><img/></a> -> <img/>.
+			// Returns the reference to <img/>
+			//
+			// @param {CKEDITOR.dom.element} link
+			// @returns {CKEDITOR.dom.element}
+			function unwrapFromLink( link ) {
+				var img = link.findOne( 'img' );
+
+				img.replace( link );
+
+				return img;
+			}
+
+			function replaceSafely( replacing, replaced ) {
+				if ( replaced.getParent() ) {
+					var range = editor.createRange();
+
+					range.moveToPosition( replaced, CKEDITOR.POSITION_BEFORE_START );
+
+					// Remove old element. Do it before insertion to avoid a case when
+					// element is moved from 'replaced' element before it, what creates
+					// a tricky case which insertElementIntorRange does not handle.
+					replaced.remove();
+
+					editable.insertElementIntoRange( replacing, range );
+				}
+				else {
+					replacing.replace( replaced );
+				}
+			}
+
+			return function( shift ) {
+				var name, i;
+
+				shift.changed = {};
+
+				for ( i = 0; i < shiftables.length; i++ ) {
+					name = shiftables[ i ];
+
+					shift.changed[ name ] = shift.oldData ?
+						shift.oldData[ name ] !== shift.newData[ name ] : false;
+				}
+
+				// Iterate over possible state variables.
+				for ( i = 0; i < shiftables.length; i++ ) {
+					name = shiftables[ i ];
+
+					stateActions[ name ]( shift,
+						shift.oldData ? shift.oldData[ name ] : null,
+						shift.newData[ name ] );
+				}
+
+				shift.inflate();
+			};
+		},
+
+		/**
+		 * Checks whether the current image ratio matches the natural one
+		 * by comparing dimensions.
+		 *
+		 * @param {CKEDITOR.dom.element} image
+		 * @returns {Boolean}
+		 */
+		checkHasNaturalRatio: function( image ) {
+			var $ = image.$,
+				natural = this.getNatural( image );
+
+			// The reason for two alternative comparisons is that the rounding can come from
+			// both dimensions, e.g. there are two cases:
+			// 	1. height is computed as a rounded relation of the real height and the value of width,
+			//	2. width is computed as a rounded relation of the real width and the value of heigh.
+			return Math.round( $.clientWidth / natural.width * natural.height ) == $.clientHeight ||
+				Math.round( $.clientHeight / natural.height * natural.width ) == $.clientWidth;
+		},
+
+		/**
+		 * Returns natural dimensions of the image. For modern browsers
+		 * it uses natural(Width|Height). For old ones (IE8) it creates
+		 * a new image and reads the dimensions.
+		 *
+		 * @param {CKEDITOR.dom.element} image
+		 * @returns {Object}
+		 */
+		getNatural: function( image ) {
+			var dimensions;
+
+			if ( image.$.naturalWidth ) {
+				dimensions = {
+					width: image.$.naturalWidth,
+					height: image.$.naturalHeight
+				};
+			} else {
+				var img = new Image();
+				img.src = image.getAttribute( 'src' );
+
+				dimensions = {
+					width: img.width,
+					height: img.height
+				};
+			}
+
+			return dimensions;
+		},
+
+		/**
+		 * Returns an attribute getter function. Default getter comes from the Link plugin
+		 * and is documented by {@link CKEDITOR.plugins.link#getLinkAttributes}.
+		 *
+		 * **Note:** It is possible to override this method and use a custom getter e.g.
+		 * in the absence of the Link plugin.
+		 *
+		 * **Note:** If a custom getter is used, a data model format it produces
+		 * must be compatible with {@link CKEDITOR.plugins.link#getLinkAttributes}.
+		 *
+		 * **Note:** A custom getter must understand the data model format produced by
+		 * {@link #getLinkAttributesParser} to work correctly.
+		 *
+		 * @returns {Function} A function that gets (composes) link attributes.
+		 * @since 4.5.5
+		 */
+		getLinkAttributesGetter: function() {
+			// https://dev.ckeditor.com/ticket/13885
+			return CKEDITOR.plugins.link.getLinkAttributes;
+		},
+
+		/**
+		 * Returns an attribute parser function. Default parser comes from the Link plugin
+		 * and is documented by {@link CKEDITOR.plugins.link#parseLinkAttributes}.
+		 *
+		 * **Note:** It is possible to override this method and use a custom parser e.g.
+		 * in the absence of the Link plugin.
+		 *
+		 * **Note:** If a custom parser is used, a data model format produced by the parser
+		 * must be compatible with {@link #getLinkAttributesGetter}.
+		 *
+		 * **Note:** If a custom parser is used, it should be compatible with the
+		 * {@link CKEDITOR.plugins.link#parseLinkAttributes} data model format. Otherwise the
+		 * Link plugin dialog may not be populated correctly with parsed data. However
+		 * as long as Enhanced Image is **not** used with the Link plugin dialog, any custom data model
+		 * will work, being stored as an internal property of Enhanced Image widget's data only.
+		 *
+		 * @returns {Function} A function that parses attributes.
+		 * @since 4.5.5
+		 */
+		getLinkAttributesParser: function() {
+			// https://dev.ckeditor.com/ticket/13885
+			return CKEDITOR.plugins.link.parseLinkAttributes;
+		}
+	};
+
+	function setWrapperAlign( widget, alignClasses ) {
+		var wrapper = widget.wrapper,
+			align = widget.data.align,
+			hasCaption = widget.data.hasCaption;
+
+		if ( alignClasses ) {
+			// Remove all align classes first.
+			for ( var i = 3; i--; )
+				wrapper.removeClass( alignClasses[ i ] );
+
+			if ( align == 'center' ) {
+				// Avoid touching non-captioned, centered widgets because
+				// they have the class set on the element instead of wrapper:
+				//
+				// 	<div class="cke_widget_wrapper">
+				// 		<p class="center-class">
+				// 			<img />
+				// 		</p>
+				// 	</div>
+				if ( hasCaption ) {
+					wrapper.addClass( alignClasses[ 1 ] );
+				}
+			} else if ( align != 'none' ) {
+				wrapper.addClass( alignClasses[ alignmentsObj[ align ] ] );
+			}
+		} else {
+			if ( align == 'center' ) {
+				if ( hasCaption )
+					wrapper.setStyle( 'text-align', 'center' );
+				else
+					wrapper.removeStyle( 'text-align' );
+
+				wrapper.removeStyle( 'float' );
+			}
+			else {
+				if ( align == 'none' )
+					wrapper.removeStyle( 'float' );
+				else
+					wrapper.setStyle( 'float', align );
+
+				wrapper.removeStyle( 'text-align' );
+			}
+		}
+	}
+
+	// Returns a function that creates widgets from all <img> and
+	// <figure class="{config.image2_captionedClass}"> elements.
+	//
+	// @param {CKEDITOR.editor} editor
+	// @returns {Function}
+	function upcastWidgetElement( editor ) {
+		var isCenterWrapper = centerWrapperChecker( editor ),
+			captionedClass = editor.config.image2_captionedClass;
+
+		// @param {CKEDITOR.htmlParser.element} el
+		// @param {Object} data
+		return function( el, data ) {
+			var dimensions = { width: 1, height: 1 },
+				name = el.name,
+				image;
+
+			// https://dev.ckeditor.com/ticket/11110 Don't initialize on pasted fake objects.
+			if ( el.attributes[ 'data-cke-realelement' ] )
+				return;
+
+			// If a center wrapper is found, there are 3 possible cases:
+			//
+			// 1. <div style="text-align:center"><figure>...</figure></div>.
+			//    In this case centering is done with a class set on widget.wrapper.
+			//    Simply replace centering wrapper with figure (it's no longer necessary).
+			//
+			// 2. <p style="text-align:center"><img/></p>.
+			//    Nothing to do here: <p> remains for styling purposes.
+			//
+			// 3. <div style="text-align:center"><img/></div>.
+			//    Nothing to do here (2.) but that case is only possible in enterMode different
+			//    than ENTER_P.
+			if ( isCenterWrapper( el ) ) {
+				if ( name == 'div' ) {
+					var figure = el.getFirst( 'figure' );
+
+					// Case #1.
+					if ( figure ) {
+						el.replaceWith( figure );
+						el = figure;
+					}
+				}
+				// Cases #2 and #3 (handled transparently)
+
+				// If there's a centering wrapper, save it in data.
+				data.align = 'center';
+
+				// Image can be wrapped in link <a><img/></a>.
+				image = el.getFirst( 'img' ) || el.getFirst( 'a' ).getFirst( 'img' );
+			}
+
+			// No center wrapper has been found.
+			else if ( name == 'figure' && el.hasClass( captionedClass ) ) {
+				image = el.find( function( child ) {
+					return child.name === 'img' &&
+						CKEDITOR.tools.array.indexOf( [ 'figure', 'a' ], child.parent.name ) !== -1;
+				}, true )[ 0 ];
+
+				// Upcast linked image like <a><img/></a>.
+			} else if ( isLinkedOrStandaloneImage( el ) ) {
+				image = el.name == 'a' ? el.children[ 0 ] : el;
+			}
+
+			if ( !image )
+				return;
+
+			// If there's an image, then cool, we got a widget.
+			// Now just remove dimension attributes expressed with %.
+			for ( var d in dimensions ) {
+				var dimension = image.attributes[ d ];
+
+				if ( dimension && dimension.match( regexPercent ) )
+					delete image.attributes[ d ];
+			}
+
+			return el;
+		};
+	}
+
+	// Returns a function which transforms the widget to the external format
+	// according to the current configuration.
+	//
+	// @param {CKEDITOR.editor}
+	function downcastWidgetElement( editor ) {
+		var alignClasses = editor.config.image2_alignClasses;
+
+		// @param {CKEDITOR.htmlParser.element} el
+		return function( el ) {
+			// In case of <a><img/></a>, <img/> is the element to hold
+			// inline styles or classes (image2_alignClasses).
+			var attrsHolder = el.name == 'a' ? el.getFirst() : el,
+				attrs = attrsHolder.attributes,
+				align = this.data.align;
+
+			// De-wrap the image from resize handle wrapper.
+			// Only block widgets have one.
+			if ( !this.inline ) {
+				var resizeWrapper = el.getFirst( 'span' );
+
+				if ( resizeWrapper )
+					resizeWrapper.replaceWith( resizeWrapper.getFirst( { img: 1, a: 1 } ) );
+			}
+
+			if ( align && align != 'none' ) {
+				var styles = CKEDITOR.tools.parseCssText( attrs.style || '' );
+
+				// When the widget is captioned (<figure>) and internally centering is done
+				// with widget's wrapper style/class, in the external data representation,
+				// <figure> must be wrapped with an element holding an style/class:
+				//
+				// 	<div style="text-align:center">
+				// 		<figure class="image" style="display:inline-block">...</figure>
+				// 	</div>
+				// or
+				// 	<div class="some-center-class">
+				// 		<figure class="image">...</figure>
+				// 	</div>
+				//
+				if ( align == 'center' && el.name == 'figure' ) {
+					el = el.wrapWith( new CKEDITOR.htmlParser.element( 'div',
+						alignClasses ? { 'class': alignClasses[ 1 ] } : { style: 'text-align:center' } ) );
+				}
+
+				// If left/right, add float style to the downcasted element.
+				else if ( align in { left: 1, right: 1 } ) {
+					if ( alignClasses )
+						attrsHolder.addClass( alignClasses[ alignmentsObj[ align ] ] );
+					else
+						styles[ 'float' ] = align;
+				}
+
+				// Update element styles.
+				if ( !alignClasses && !CKEDITOR.tools.isEmpty( styles ) )
+					attrs.style = CKEDITOR.tools.writeCssText( styles );
+			}
+
+			return el;
+		};
+	}
+
+	// Returns a function that checks if an element is a centering wrapper.
+	//
+	// @param {CKEDITOR.editor} editor
+	// @returns {Function}
+	function centerWrapperChecker( editor ) {
+		var captionedClass = editor.config.image2_captionedClass,
+			alignClasses = editor.config.image2_alignClasses,
+			validChildren = { figure: 1, a: 1, img: 1 };
+
+		return function( el ) {
+			// Wrapper must be either <div> or <p>.
+			if ( !( el.name in { div: 1, p: 1 } ) )
+				return false;
+
+			var children = el.children;
+
+			// Centering wrapper can have only one child.
+			if ( children.length !== 1 )
+				return false;
+
+			var child = children[ 0 ];
+
+			// Only <figure> or <img /> can be first (only) child of centering wrapper,
+			// regardless of its type.
+			if ( !( child.name in validChildren ) )
+				return false;
+
+			// If centering wrapper is <p>, only <img /> can be the child.
+			//   <p style="text-align:center"><img /></p>
+			if ( el.name == 'p' ) {
+				if ( !isLinkedOrStandaloneImage( child ) )
+					return false;
+			}
+			// Centering <div> can hold <img/> or <figure>, depending on enterMode.
+			else {
+				// If a <figure> is the first (only) child, it must have a class.
+				//   <div style="text-align:center"><figure>...</figure><div>
+				if ( child.name == 'figure' ) {
+					if ( !child.hasClass( captionedClass ) )
+						return false;
+				} else {
+					// Centering <div> can hold <img/> or <a><img/></a> only when enterMode
+					// is ENTER_(BR|DIV).
+					//   <div style="text-align:center"><img /></div>
+					//   <div style="text-align:center"><a><img /></a></div>
+					if ( editor.enterMode == CKEDITOR.ENTER_P )
+						return false;
+
+					// Regardless of enterMode, a child which is not <figure> must be
+					// either <img/> or <a><img/></a>.
+					if ( !isLinkedOrStandaloneImage( child ) )
+						return false;
+				}
+			}
+
+			// Centering wrapper got to be... centering. If image2_alignClasses are defined,
+			// check for centering class. Otherwise, check the style.
+			if ( alignClasses ? el.hasClass( alignClasses[ 1 ] ) :
+					CKEDITOR.tools.parseCssText( el.attributes.style || '', true )[ 'text-align' ] == 'center' )
+				return true;
+
+			return false;
+		};
+	}
+
+	// Checks whether element is <img/> or <a><img/></a>.
+	//
+	// @param {CKEDITOR.htmlParser.element}
+	function isLinkedOrStandaloneImage( el ) {
+		if ( el.name == 'img' )
+			return true;
+		else if ( el.name == 'a' )
+			return el.children.length == 1 && el.getFirst( 'img' );
+
+		return false;
+	}
+
+	// Sets width and height of the widget image according to current widget data.
+	//
+	// @param {CKEDITOR.plugins.widget} widget
+	function setDimensions( widget ) {
+		var data = widget.data,
+			dimensions = { width: data.width, height: data.height },
+			image = widget.parts.image;
+
+		for ( var d in dimensions ) {
+			if ( dimensions[ d ] )
+				image.setAttribute( d, dimensions[ d ] );
+			else
+				image.removeAttribute( d );
+		}
+	}
+
+	// Defines all features related to drag-driven image resizing.
+	//
+	// @param {CKEDITOR.plugins.widget} widget
+	function setupResizer( widget ) {
+		var editor = widget.editor,
+			editable = editor.editable(),
+			doc = editor.document,
+
+			// Store the resizer in a widget for testing (https://dev.ckeditor.com/ticket/11004).
+			resizer = widget.resizer = doc.createElement( 'span' );
+
+		resizer.addClass( 'cke_image_resizer' );
+		resizer.setAttribute( 'title', editor.lang.image2.resizer );
+		resizer.append( new CKEDITOR.dom.text( '\u200b', doc ) );
+
+		// Inline widgets don't need a resizer wrapper as an image spans the entire widget.
+		if ( !widget.inline ) {
+			var imageOrLink = widget.parts.link || widget.parts.image,
+				oldResizeWrapper = imageOrLink.getParent(),
+				resizeWrapper = doc.createElement( 'span' );
+
+			resizeWrapper.addClass( 'cke_image_resizer_wrapper' );
+			resizeWrapper.append( imageOrLink );
+			resizeWrapper.append( resizer );
+			widget.element.append( resizeWrapper, true );
+
+			// Remove the old wrapper which could came from e.g. pasted HTML
+			// and which could be corrupted (e.g. resizer span has been lost).
+			if ( oldResizeWrapper.is( 'span' ) )
+				oldResizeWrapper.remove();
+		} else {
+			widget.wrapper.append( resizer );
+		}
+
+		// Calculate values of size variables and mouse offsets.
+		resizer.on( 'mousedown', function( evt ) {
+			var image = widget.parts.image,
+
+				// Don't update attributes if less than 15.
+				// This is to prevent images to visually disappear.
+				min = {
+					width: 15,
+					height: 15
+				},
+
+				max = getMaxSize(),
+
+				// "factor" can be either 1 or -1. I.e.: For right-aligned images, we need to
+				// subtract the difference to get proper width, etc. Without "factor",
+				// resizer starts working the opposite way.
+				factor = widget.data.align == 'right' ? -1 : 1,
+
+				// The x-coordinate of the mouse relative to the screen
+				// when button gets pressed.
+				startX = evt.data.$.screenX,
+				startY = evt.data.$.screenY,
+
+				// The initial dimensions and aspect ratio of the image.
+				startWidth = image.$.clientWidth,
+				startHeight = image.$.clientHeight,
+				ratio = startWidth / startHeight,
+
+				listeners = [],
+
+				// A class applied to editable during resizing.
+				cursorClass = 'cke_image_s' + ( !~factor ? 'w' : 'e' ),
+
+				nativeEvt, newWidth, newHeight, updateData,
+				moveDiffX, moveDiffY, moveRatio;
+
+			// Save the undo snapshot first: before resizing.
+			editor.fire( 'saveSnapshot' );
+
+			// Mousemove listeners are removed on mouseup.
+			attachToDocuments( 'mousemove', onMouseMove, listeners );
+
+			// Clean up the mousemove listener. Update widget data if valid.
+			attachToDocuments( 'mouseup', onMouseUp, listeners );
+
+			// The entire editable will have the special cursor while resizing goes on.
+			editable.addClass( cursorClass );
+
+			// This is to always keep the resizer element visible while resizing.
+			resizer.addClass( 'cke_image_resizing' );
+
+			// Attaches an event to a global document if inline editor.
+			// Additionally, if classic (`iframe`-based) editor, also attaches the same event to `iframe`'s document.
+			function attachToDocuments( name, callback, collection ) {
+				var globalDoc = CKEDITOR.document,
+					listeners = [];
+
+				if ( !doc.equals( globalDoc ) )
+					listeners.push( globalDoc.on( name, callback ) );
+
+				listeners.push( doc.on( name, callback ) );
+
+				if ( collection ) {
+					for ( var i = listeners.length; i--; )
+						collection.push( listeners.pop() );
+				}
+			}
+
+			// Calculate with first, and then adjust height, preserving ratio.
+			function adjustToX() {
+				newWidth = startWidth + factor * moveDiffX;
+				newHeight = Math.round( newWidth / ratio );
+			}
+
+			// Calculate height first, and then adjust width, preserving ratio.
+			function adjustToY() {
+				newHeight = startHeight - moveDiffY;
+				newWidth = Math.round( newHeight * ratio );
+			}
+
+			// This is how variables refer to the geometry.
+			// Note: x corresponds to moveOffset, this is the position of mouse
+			// Note: o corresponds to [startX, startY].
+			//
+			// 	+--------------+--------------+
+			// 	|              |              |
+			// 	|      I       |      II      |
+			// 	|              |              |
+			// 	+------------- o -------------+ _ _ _
+			// 	|              |              |      ^
+			// 	|      VI      |     III      |      | moveDiffY
+			// 	|              |         x _ _ _ _ _ v
+			// 	+--------------+---------|----+
+			// 	               |         |
+			// 	                <------->
+			// 	                moveDiffX
+			function onMouseMove( evt ) {
+				nativeEvt = evt.data.$;
+
+				// This is how far the mouse is from the point the button was pressed.
+				moveDiffX = nativeEvt.screenX - startX;
+				moveDiffY = startY - nativeEvt.screenY;
+
+				// This is the aspect ratio of the move difference.
+				moveRatio = Math.abs( moveDiffX / moveDiffY );
+
+				// Left, center or none-aligned widget.
+				if ( factor == 1 ) {
+					if ( moveDiffX <= 0 ) {
+						// Case: IV.
+						if ( moveDiffY <= 0 )
+							adjustToX();
+
+						// Case: I.
+						else {
+							if ( moveRatio >= ratio )
+								adjustToX();
+							else
+								adjustToY();
+						}
+					} else {
+						// Case: III.
+						if ( moveDiffY <= 0 ) {
+							if ( moveRatio >= ratio )
+								adjustToY();
+							else
+								adjustToX();
+						}
+
+						// Case: II.
+						else {
+							adjustToY();
+						}
+					}
+				}
+
+				// Right-aligned widget. It mirrors behaviours, so I becomes II,
+				// IV becomes III and vice-versa.
+				else {
+					if ( moveDiffX <= 0 ) {
+						// Case: IV.
+						if ( moveDiffY <= 0 ) {
+							if ( moveRatio >= ratio )
+								adjustToY();
+							else
+								adjustToX();
+						}
+
+						// Case: I.
+						else {
+							adjustToY();
+						}
+					} else {
+						// Case: III.
+						if ( moveDiffY <= 0 )
+							adjustToX();
+
+						// Case: II.
+						else {
+							if ( moveRatio >= ratio ) {
+								adjustToX();
+							} else {
+								adjustToY();
+							}
+						}
+					}
+				}
+
+				if ( isAllowedSize( newWidth, newHeight ) ) {
+					updateData = { width: newWidth, height: newHeight };
+					image.setAttributes( updateData );
+				}
+			}
+
+			function onMouseUp() {
+				var l;
+
+				while ( ( l = listeners.pop() ) )
+					l.removeListener();
+
+				// Restore default cursor by removing special class.
+				editable.removeClass( cursorClass );
+
+				// This is to bring back the regular behaviour of the resizer.
+				resizer.removeClass( 'cke_image_resizing' );
+
+				if ( updateData ) {
+					widget.setData( updateData );
+
+					// Save another undo snapshot: after resizing.
+					editor.fire( 'saveSnapshot' );
+				}
+
+				// Don't update data twice or more.
+				updateData = false;
+			}
+
+			function getMaxSize() {
+				var maxSize = editor.config.image2_maxSize,
+					natural;
+
+				if ( !maxSize ) {
+					return null;
+				}
+
+				maxSize = CKEDITOR.tools.copy( maxSize );
+				natural = CKEDITOR.plugins.image2.getNatural( image );
+
+				maxSize.width = Math.max( maxSize.width === 'natural' ? natural.width : maxSize.width, min.width );
+				maxSize.height = Math.max( maxSize.height === 'natural' ? natural.height : maxSize.height, min.width );
+
+				return maxSize;
+			}
+
+			function isAllowedSize( width, height ) {
+				var isTooSmall = width < min.width || height < min.height,
+					isTooBig = max && ( width > max.width || height > max.height );
+				return !isTooSmall && !isTooBig;
+			}
+		} );
+
+		// Change the position of the widget resizer when data changes.
+		widget.on( 'data', function() {
+			resizer[ widget.data.align == 'right' ? 'addClass' : 'removeClass' ]( 'cke_image_resizer_left' );
+		} );
+	}
+
+	// Integrates widget alignment setting with justify
+	// plugin's commands (execution and refreshment).
+	// @param {CKEDITOR.editor} editor
+	// @param {String} value 'left', 'right', 'center' or 'block'
+	function alignCommandIntegrator( editor ) {
+		var execCallbacks = [],
+			enabled;
+
+		return function( value ) {
+			var command = editor.getCommand( 'justify' + value );
+
+			// Most likely, the justify plugin isn't loaded.
+			if ( !command )
+				return;
+
+			// This command will be manually refreshed along with
+			// other commands after exec.
+			execCallbacks.push( function() {
+				command.refresh( editor, editor.elementPath() );
+			} );
+
+			if ( value in { right: 1, left: 1, center: 1 } ) {
+				command.on( 'exec', function( evt ) {
+					var widget = getFocusedWidget( editor );
+
+					if ( widget ) {
+						widget.setData( 'align', value );
+
+						// Once the widget changed its align, all the align commands
+						// must be refreshed: the event is to be cancelled.
+						for ( var i = execCallbacks.length; i--; )
+							execCallbacks[ i ]();
+
+						evt.cancel();
+					}
+				} );
+			}
+
+			command.on( 'refresh', function( evt ) {
+				var widget = getFocusedWidget( editor ),
+					allowed = { right: 1, left: 1, center: 1 };
+
+				if ( !widget )
+					return;
+
+				// Cache "enabled" on first use. This is because filter#checkFeature may
+				// not be available during plugin's afterInit in the future — a moment when
+				// alignCommandIntegrator is called.
+				if ( enabled === undefined )
+					enabled = editor.filter.checkFeature( editor.widgets.registered.image.features.align );
+
+				// Don't allow justify commands when widget alignment is disabled (https://dev.ckeditor.com/ticket/11004).
+				if ( !enabled )
+					this.setState( CKEDITOR.TRISTATE_DISABLED );
+				else {
+					this.setState(
+						( widget.data.align == value ) ? (
+							CKEDITOR.TRISTATE_ON
+						) : (
+							( value in allowed ) ? CKEDITOR.TRISTATE_OFF : CKEDITOR.TRISTATE_DISABLED
+						)
+					);
+				}
+
+				evt.cancel();
+			} );
+		};
+	}
+
+	function linkCommandIntegrator( editor ) {
+		// Nothing to integrate with if link is not loaded.
+		if ( !editor.plugins.link )
+			return;
+
+		var listener = CKEDITOR.on( 'dialogDefinition', function( evt ) {
+			var dialog = evt.data;
+
+			if ( dialog.name == 'link' ) {
+				var def = dialog.definition;
+
+				var onShow = def.onShow,
+					onOk = def.onOk;
+
+				def.onShow = function() {
+					var widget = getFocusedWidget( editor ),
+						displayTextField = this.getContentElement( 'info', 'linkDisplayText' ).getElement().getParent().getParent();
+
+					// Widget cannot be enclosed in a link, i.e.
+					//		<a>foo<inline widget/>bar</a>
+					if ( widget && ( widget.inline ? !widget.wrapper.getAscendant( 'a' ) : 1 ) ) {
+						this.setupContent( widget.data.link || {} );
+
+						// Hide the display text in case of linking image2 widget.
+						displayTextField.hide();
+					} else {
+						// Make sure that display text is visible, as it might be hidden by image2 integration
+						// before.
+						displayTextField.show();
+						onShow.apply( this, arguments );
+					}
+				};
+
+				// Set widget data if linking the widget using
+				// link dialog (instead of default action).
+				// State shifter handles data change and takes
+				// care of internal DOM structure of linked widget.
+				def.onOk = function() {
+					var widget = getFocusedWidget( editor );
+
+					// Widget cannot be enclosed in a link, i.e.
+					//		<a>foo<inline widget/>bar</a>
+					if ( widget && ( widget.inline ? !widget.wrapper.getAscendant( 'a' ) : 1 ) ) {
+						var data = {};
+
+						// Collect data from fields.
+						this.commitContent( data );
+
+						// Set collected data to widget.
+						widget.setData( 'link', data );
+					} else {
+						onOk.apply( this, arguments );
+					}
+				};
+			}
+		} );
+		// Listener has to be removed due to leaking the editor reference (#589).
+		editor.on( 'destroy', function() {
+			listener.removeListener();
+		} );
+
+		// Overwrite the default behavior of unlink command.
+		editor.getCommand( 'unlink' ).on( 'exec', function( evt ) {
+			var widget = getFocusedWidget( editor );
+
+			// Override unlink only when link truly belongs to the widget.
+			// If wrapped inline widget in a link, let default unlink work (https://dev.ckeditor.com/ticket/11814).
+			if ( !widget || !widget.parts.link )
+				return;
+
+			widget.setData( 'link', null );
+
+			// Selection (which is fake) may not change if unlinked image in focused widget,
+			// i.e. if captioned image. Let's refresh command state manually here.
+			this.refresh( editor, editor.elementPath() );
+
+			evt.cancel();
+		} );
+
+		// Overwrite default refresh of unlink command.
+		editor.getCommand( 'unlink' ).on( 'refresh', function( evt ) {
+			var widget = getFocusedWidget( editor );
+
+			if ( !widget )
+				return;
+
+			// Note that widget may be wrapped in a link, which
+			// does not belong to that widget (https://dev.ckeditor.com/ticket/11814).
+			this.setState( widget.data.link || widget.wrapper.getAscendant( 'a' ) ?
+				CKEDITOR.TRISTATE_OFF : CKEDITOR.TRISTATE_DISABLED );
+
+			evt.cancel();
+		} );
+	}
+
+	// Returns the focused widget, if of the type specific for this plugin.
+	// If no widget is focused, `null` is returned.
+	//
+	// @param {CKEDITOR.editor}
+	// @returns {CKEDITOR.plugins.widget}
+	function getFocusedWidget( editor ) {
+		var widget = editor.widgets.focused;
+
+		if ( widget && widget.name == 'image' )
+			return widget;
+
+		return null;
+	}
+
+	// Returns a set of widget allowedContent rules, depending
+	// on configurations like config#image2_alignClasses or
+	// config#image2_captionedClass.
+	//
+	// @param {CKEDITOR.editor}
+	// @returns {Object}
+	function getWidgetAllowedContent( editor ) {
+		var alignClasses = editor.config.image2_alignClasses,
+			rules = {
+				// Widget may need <div> or <p> centering wrapper.
+				div: {
+					match: centerWrapperChecker( editor )
+				},
+				p: {
+					match: centerWrapperChecker( editor )
+				},
+				img: {
+					attributes: '!src,alt,width,height'
+				},
+				figure: {
+					classes: '!' + editor.config.image2_captionedClass
+				},
+				figcaption: true
+			};
+
+		if ( alignClasses ) {
+			// Centering class from the config.
+			rules.div.classes = alignClasses[ 1 ];
+			rules.p.classes = rules.div.classes;
+
+			// Left/right classes from the config.
+			rules.img.classes = alignClasses[ 0 ] + ',' + alignClasses[ 2 ];
+			rules.figure.classes += ',' + rules.img.classes;
+		} else {
+			// Centering with text-align.
+			rules.div.styles = 'text-align';
+			rules.p.styles = 'text-align';
+
+			rules.img.styles = 'float';
+			rules.figure.styles = 'float,display';
+		}
+
+		return rules;
+	}
+
+	// Returns a set of widget feature rules, depending
+	// on editor configuration. Note that the following may not cover
+	// all the possible cases since requiredContent supports a single
+	// tag only.
+	//
+	// @param {CKEDITOR.editor}
+	// @returns {Object}
+	function getWidgetFeatures( editor ) {
+		var alignClasses = editor.config.image2_alignClasses,
+			features = {
+				dimension: {
+					requiredContent: 'img[width,height]'
+				},
+				align: {
+					requiredContent: 'img' +
+						( alignClasses ? '(' + alignClasses[ 0 ] + ')' : '{float}' )
+				},
+				caption: {
+					requiredContent: 'figcaption'
+				}
+			};
+
+		return features;
+	}
+
+	// Returns element which is styled, considering current
+	// state of the widget.
+	//
+	// @see CKEDITOR.plugins.widget#applyStyle
+	// @param {CKEDITOR.plugins.widget} widget
+	// @returns {CKEDITOR.dom.element}
+	function getStyleableElement( widget ) {
+		return widget.data.hasCaption ? widget.element : widget.parts.image;
+	}
+} )();
+
+/**
+ * A CSS class applied to the `<figure>` element of a captioned image.
+ *
+ * Read more in the {@glink features/image2 documentation} and see the
+ * {@glink examples/image2 example}.
+ *
+ *		// Changes the class to "captionedImage".
+ *		config.image2_captionedClass = 'captionedImage';
+ *
+ * @cfg {String} [image2_captionedClass='image']
+ * @member CKEDITOR.config
+ */
+CKEDITOR.config.image2_captionedClass = 'image';
+
+/**
+ * Determines whether dimension inputs should be automatically filled when the image URL changes in the Enhanced Image
+ * plugin dialog window.
+ *
+ * Read more in the {@glink features/image2 documentation} and see the
+ * {@glink examples/image2 example}.
+ *
+ *		config.image2_prefillDimensions = false;
+ *
+ * @since 4.5.0
+ * @cfg {Boolean} [image2_prefillDimensions=true]
+ * @member CKEDITOR.config
+ */
+
+/**
+ * Disables the image resizer. By default the resizer is enabled.
+ *
+ * Read more in the {@glink features/image2 documentation} and see the
+ * {@glink examples/image2 example}.
+ *
+ *		config.image2_disableResizer = true;
+ *
+ * @since 4.5.0
+ * @cfg {Boolean} [image2_disableResizer=false]
+ * @member CKEDITOR.config
+ */
+
+/**
+ * CSS classes applied to aligned images. Useful to take control over the way
+ * the images are aligned, i.e. to customize output HTML and integrate external stylesheets.
+ *
+ * Classes should be defined in an array of three elements, containing left, center, and right
+ * alignment classes, respectively. For example:
+ *
+ *		config.image2_alignClasses = [ 'align-left', 'align-center', 'align-right' ];
+ *
+ * **Note**: Once this configuration option is set, the plugin will no longer produce inline
+ * styles for alignment. It means that e.g. the following HTML will be produced:
+ *
+ *		<img alt="My image" class="custom-center-class" src="foo.png" />
+ *
+ * instead of:
+ *
+ *		<img alt="My image" style="float:left" src="foo.png" />
+ *
+ * **Note**: Once this configuration option is set, corresponding style definitions
+ * must be supplied to the editor:
+ *
+ * * For {@glink guide/dev_framed classic editor} it can be done by defining additional
+ * styles in the {@link CKEDITOR.config#contentsCss stylesheets loaded by the editor}. The same
+ * styles must be provided on the target page where the content will be loaded.
+ * * For {@glink guide/dev_inline inline editor} the styles can be defined directly
+ * with `<style> ... <style>` or `<link href="..." rel="stylesheet">`, i.e. within the `<head>`
+ * of the page.
+ *
+ * For example, considering the following configuration:
+ *
+ *		config.image2_alignClasses = [ 'align-left', 'align-center', 'align-right' ];
+ *
+ * CSS rules can be defined as follows:
+ *
+ *		.align-left {
+ *			float: left;
+ *		}
+ *
+ *		.align-right {
+ *			float: right;
+ *		}
+ *
+ *		.align-center {
+ *			text-align: center;
+ *		}
+ *
+ *		.align-center > figure {
+ *			display: inline-block;
+ *		}
+ *
+ * Read more in the {@glink features/image2 documentation} and see the
+ * {@glink examples/image2 example}.
+ *
+ * @since 4.4.0
+ * @cfg {String[]} [image2_alignClasses=null]
+ * @member CKEDITOR.config
+ */
+
+/**
+ * Determines whether alternative text is required for the captioned image.
+ *
+ *		config.image2_altRequired = true;
+ *
+ * Read more in the {@glink features/image2 documentation} and see the
+ * {@glink examples/image2 example}.
+ *
+ * @since 4.6.0
+ * @cfg {Boolean} [image2_altRequired=false]
+ * @member CKEDITOR.config
+ */
+
+/**
+ * Determines the maximum size that an image can be resized to with the resize handle.
+ *
+ * It stores two properties: `width` and `height`. They can be set with one of the two types:
+ *
+ * * A number representing a value that limits the maximum size in pixel units:
+ *
+ * ```js
+ *	config.image2_maxSize = {
+ *		height: 300,
+ *		width: 250
+ *	};
+ * ```
+ *
+ * * A string representing the natural image size, so each image resize operation is limited to its own natural height or width:
+ *
+ * ```js
+ *	config.image2_maxSize = {
+ *		height: 'natural',
+ *		width: 'natural'
+ *	}
+ * ```
+ *
+ * Note: An image can still be resized to bigger dimensions when using the image dialog.
+ *
+ * @since 4.12.0
+ * @cfg {Object.<String, Number/String>} [image2_maxSize]
+ * @member CKEDITOR.config
+ */

BIN
public/ckeditor/plugins/image2/samples/assets/image1.jpg


BIN
public/ckeditor/plugins/image2/samples/assets/image2.jpg


File diff suppressed because it is too large
+ 44 - 0
public/ckeditor/plugins/image2/samples/image2.html


+ 1 - 0
public/ckeditor/plugins/pastebase64/plugin.js

@@ -65,6 +65,7 @@
       // position
       setTimeout(function() {
         editor.insertElement(element);
+        editor.setData(editor.getData());
       }, 10);
     };
 

+ 6 - 3
src/components/ckeditor.vue

@@ -54,9 +54,9 @@ export default {
     }
   },
   mounted() {
-    var removePluginStr = "bidi";
+    var removePluginStr = "bidi,image";
     var removeButtonStr =
-      "Image,Styles,Format,ShowBlocks,Iframe,PageBreak,Smiley,Flash,Language,JustifyBlock,JustifyRight,JustifyCenter,JustifyLeft,CreateDiv,CopyFormatting,ImageButton,Button,HiddenField,Select,Textarea,TextField,Radio,Checkbox,Form,BGColor,SelectAll,Replace,Find,Templates,Print,Preview,NewPage,Save,Underline,Subscript,Superscript,HorizontalRule,Unlink,Link,Scayt,Cut,Copy,Paste,PasteText,PasteFromWord,Maximize,Italic,Bold,NumberedList,BulletedList,Indent,Outdent,Blockquote,About,RemoveFormat,Strike";
+      "Styles,Format,ShowBlocks,Iframe,PageBreak,Smiley,Flash,Language,JustifyBlock,JustifyRight,JustifyCenter,JustifyLeft,CreateDiv,CopyFormatting,ImageButton,Button,HiddenField,Select,Textarea,TextField,Radio,Checkbox,Form,BGColor,SelectAll,Replace,Find,Templates,Print,Preview,NewPage,Save,Underline,Subscript,Superscript,HorizontalRule,Unlink,Link,Scayt,Cut,Copy,Paste,PasteText,PasteFromWord,Maximize,Italic,Bold,NumberedList,BulletedList,Indent,Outdent,Blockquote,About,RemoveFormat,Strike";
     if (!this.extrabuttons || this.extrabuttons.indexOf("Font") == -1) {
       removeButtonStr = removeButtonStr + ",Font";
     }
@@ -72,7 +72,7 @@ export default {
       language: this.language,
       height: this.height,
       width: this.width,
-      extraPlugins: this.extraplugins + ",base64image,pastebase64",
+      extraPlugins: this.extraplugins + ",image2,base64image,pastebase64",
       removeButtons: removeButtonStr,
       removePlugins: removePluginStr
     };
@@ -101,4 +101,7 @@ export default {
   display: table;
   clear: both;
 }
+.cke_button__image {
+  display: none !important;
+}
 </style>

+ 2 - 0
src/modules/basic/view/course.vue

@@ -410,12 +410,14 @@
         </el-dialog>
 
         <!-- 页面列表 -->
+        <!-- FIXME: element-ui style bug https://github.com/ElemeFE/element/issues/16167 -->
         <el-table
           :data="tableData"
           border
           resizable
           stripe
           @selection-change="selectChange"
+          style="width: 99.99% !important;"
         >
           <el-table-column type="selection" width="40" />
           <el-table-column prop="id" label="课程ID" width="80" />

+ 13 - 9
src/modules/basic/view/school_config.vue

@@ -51,15 +51,11 @@
             off-text="否"
           ></el-switch>
         </el-form-item>
-        <el-form-item
-          label="展示APP下载二维码"
-          prop="SHOW_STUDENT_CLIENT_APP_QRCODE"
-        >
-          <el-switch
-            v-model="ruleForm.SHOW_STUDENT_CLIENT_APP_QRCODE"
-            on-text="是"
-            off-text="否"
-          ></el-switch>
+        <el-form-item label="证件号隐私模式" prop="ID_NUMBER_PRIVATE_MODE">
+          <el-switch v-model="ruleForm.ID_NUMBER_PRIVATE_MODE" on-text="是" off-text="否"></el-switch>
+        </el-form-item>
+        <el-form-item label="展示APP下载二维码" prop="SHOW_STUDENT_CLIENT_APP_QRCODE">
+          <el-switch v-model="ruleForm.SHOW_STUDENT_CLIENT_APP_QRCODE" on-text="是" off-text="否"></el-switch>
         </el-form-item>
         <el-form-item label="开放APP" prop="APP_ENABLED">
           <el-switch
@@ -128,6 +124,7 @@ export default {
         SHOW_STUDENT_CLIENT_APP_QRCODE: false,
         STUDENT_CLIENT_THIRD_PARTY_LOGIN_URL: "",
         SHOW_QMTH_LOGO: false,
+        ID_NUMBER_PRIVATE_MODE: false,
         APP_ENABLED: false,
         WEIXIN_ANSWER_ENABLED: false,
         IDENTIFICATION_OF_LIVING_BODY_SCHEME: "",
@@ -136,6 +133,7 @@ export default {
           SHOW_STUDENT_CLIENT_APP_QRCODE: false,
           STUDENT_CLIENT_THIRD_PARTY_LOGIN_URL: "",
           SHOW_QMTH_LOGO: false,
+          ID_NUMBER_PRIVATE_MODE: false,
           APP_ENABLED: false,
           WEIXIN_ANSWER_ENABLED: false,
           IDENTIFICATION_OF_LIVING_BODY_SCHEME: ""
@@ -158,6 +156,7 @@ export default {
           this.ruleForm.properties.STUDENT_CLIENT_ACCESS_FROM_THIRD_PARTY = this.ruleForm.STUDENT_CLIENT_ACCESS_FROM_THIRD_PARTY;
           this.ruleForm.properties.STUDENT_CLIENT_THIRD_PARTY_LOGIN_URL = this.ruleForm.STUDENT_CLIENT_THIRD_PARTY_LOGIN_URL;
           this.ruleForm.properties.SHOW_QMTH_LOGO = this.ruleForm.SHOW_QMTH_LOGO;
+          this.ruleForm.properties.ID_NUMBER_PRIVATE_MODE = this.ruleForm.ID_NUMBER_PRIVATE_MODE;
           this.ruleForm.properties.APP_ENABLED = this.ruleForm.APP_ENABLED;
           this.ruleForm.properties.WEIXIN_ANSWER_ENABLED = this.ruleForm.WEIXIN_ANSWER_ENABLED;
           this.ruleForm.properties.SHOW_STUDENT_CLIENT_APP_QRCODE = this.ruleForm.SHOW_STUDENT_CLIENT_APP_QRCODE;
@@ -197,6 +196,9 @@ export default {
           this.ruleForm.STUDENT_CLIENT_THIRD_PARTY_LOGIN_URL =
             response.data.STUDENT_CLIENT_THIRD_PARTY_LOGIN_URL;
           this.ruleForm.SHOW_QMTH_LOGO = response.data.SHOW_QMTH_LOGO == "true";
+          this.ruleForm.ID_NUMBER_PRIVATE_MODE =
+            response.data.ID_NUMBER_PRIVATE_MODE == "true";
+
           this.ruleForm.APP_ENABLED = response.data.APP_ENABLED == "true";
           this.ruleForm.WEIXIN_ANSWER_ENABLED =
             response.data.WEIXIN_ANSWER_ENABLED == "true";
@@ -249,6 +251,8 @@ export default {
             newForm.STUDENT_CLIENT_THIRD_PARTY_LOGIN_URL ==
               this.originalRuleForm.STUDENT_CLIENT_THIRD_PARTY_LOGIN_URL &&
             newForm.SHOW_QMTH_LOGO == this.originalRuleForm.SHOW_QMTH_LOGO &&
+            newForm.ID_NUMBER_PRIVATE_MODE ==
+              this.originalRuleForm.ID_NUMBER_PRIVATE_MODE &&
             newForm.APP_ENABLED == this.originalRuleForm.APP_ENABLED &&
             newForm.WEIXIN_ANSWER_ENABLED ==
               this.originalRuleForm.WEIXIN_ANSWER_ENABLED &&

+ 2 - 2
src/modules/examwork/view/examInfo.vue

@@ -210,8 +210,8 @@
                     </el-dropdown-item>
                     <el-dropdown-item>
                       <el-button
-                        :disabled="
-                          !(
+                        v-show="
+                          (
                             rolePrivileges.update_exam &&
                             (scope.row.examType == 'OFFLINE' ||
                               scope.row.examType == 'ONLINE') &&

+ 68 - 248
src/modules/examwork/view/examStudent.vue

@@ -20,30 +20,17 @@
                 :label="item.name"
                 :value="item.id"
                 :key="item.id"
-              >
-              </el-option>
+              ></el-option>
             </el-select>
           </el-form-item>
           <el-form-item label="姓名">
-            <el-input
-              class="input"
-              placeholder="请输入姓名"
-              v-model="formSearch.studentName"
-            ></el-input>
+            <el-input class="input" placeholder="请输入姓名" v-model="formSearch.studentName"></el-input>
           </el-form-item>
           <el-form-item label="学号">
-            <el-input
-              class="input"
-              placeholder="请输入学号"
-              v-model="formSearch.studentCode"
-            ></el-input>
+            <el-input class="input" placeholder="请输入学号" v-model="formSearch.studentCode"></el-input>
           </el-form-item>
           <el-form-item label="专业">
-            <el-input
-              class="input"
-              placeholder="请输入专业"
-              v-model="formSearch.specialtyName"
-            ></el-input>
+            <el-input class="input" placeholder="请输入专业" v-model="formSearch.specialtyName"></el-input>
           </el-form-item>
           <el-form-item label="课程">
             <el-select
@@ -61,8 +48,7 @@
                 :label="item.name + ' - ' + item.code"
                 :value="item.id"
                 :key="item.id"
-              >
-              </el-option>
+              ></el-option>
             </el-select>
           </el-form-item>
           <el-form-item label="学习中心">
@@ -82,30 +68,17 @@
                 :label="item.name + ' - ' + item.code"
                 :value="item.id"
                 :key="item.id"
-              >
-              </el-option>
+              ></el-option>
             </el-select>
           </el-form-item>
           <el-form-item label="采集人">
-            <el-input
-              class="input"
-              placeholder="请输入采集人"
-              v-model="formSearch.infoCollector"
-            ></el-input>
+            <el-input class="input" placeholder="请输入采集人" v-model="formSearch.infoCollector"></el-input>
           </el-form-item>
           <el-form-item label="身份证">
-            <el-input
-              class="input"
-              placeholder="请输入身份证"
-              v-model="formSearch.identityNumber"
-            ></el-input>
+            <el-input class="input" placeholder="请输入身份证" v-model="formSearch.identityNumber"></el-input>
           </el-form-item>
           <el-form-item label="考点">
-            <el-input
-              class="input"
-              placeholder="请输入考点"
-              v-model="formSearch.examSite"
-            ></el-input>
+            <el-input class="input" placeholder="请输入考点" v-model="formSearch.examSite"></el-input>
           </el-form-item>
           <el-form-item class="d-block">
             <el-button
@@ -114,39 +87,29 @@
               type="primary"
               icon="el-icon-search"
               @click="resetPageAndSearchForm"
-              >查询
-            </el-button>
-            <el-button
-              size="small"
-              icon="el-icon-refresh"
-              @click="resetSearchForm"
-            >
-              重置
-            </el-button>
+            >查询</el-button>
+            <el-button size="small" icon="el-icon-refresh" @click="resetSearchForm">重置</el-button>
             <el-button
               v-if="rolePrivileges.add_examStudent"
               size="small"
               type="primary"
               icon="el-icon-plus"
               @click="openAddingDialog"
-              >新增
-            </el-button>
+            >新增</el-button>
             <el-button
               v-if="rolePrivileges.del_examStudent"
               size="small"
               type="danger"
               icon="el-icon-delete"
               @click="deleteStuByExam"
-              >批次删除
-            </el-button>
+            >批次删除</el-button>
             <el-button
               v-if="rolePrivileges.copy_examStudent"
               size="small"
               type="primary"
               icon="el-icon-document"
               @click="copy"
-              >复制
-            </el-button>
+            >复制</el-button>
           </el-form-item>
         </el-form>
 
@@ -160,8 +123,7 @@
             icon="el-icon-delete"
             @click="deleteStuBatch"
             :disabled="noBatchSelected"
-            >删除
-          </el-button>
+          >删除</el-button>
         </div>
 
         <div style="width: 100%;margin-bottom: 10px;"></div>
@@ -198,8 +160,7 @@
                   :value="item.id"
                   :key="item.id"
                   :disabled="!item.enable"
-                >
-                </el-option>
+                ></el-option>
               </el-select>
             </el-form-item>
             <el-form-item label="姓名" prop="studentName">
@@ -244,8 +205,7 @@
                   :value="item.id"
                   :key="item.id"
                   :disabled="!item.enable"
-                >
-                </el-option>
+                ></el-option>
               </el-select>
             </el-form-item>
             <el-form-item label="课程" prop="courseId">
@@ -265,23 +225,12 @@
                   :value="item.id"
                   :key="item.id"
                   :disabled="!item.enable"
-                >
-                </el-option>
+                ></el-option>
               </el-select>
             </el-form-item>
             <el-form-item label="试卷类型" prop="paperType">
-              <el-select
-                class="input"
-                v-model="examStudentForm.paperType"
-                placeholder="请选择"
-              >
-                <el-option
-                  v-for="item in paperTypeList"
-                  :label="item"
-                  :value="item"
-                  :key="item"
-                >
-                </el-option>
+              <el-select class="input" v-model="examStudentForm.paperType" placeholder="请选择">
+                <el-option v-for="item in paperTypeList" :label="item" :value="item" :key="item"></el-option>
               </el-select>
             </el-form-item>
             <el-form-item label="专业">
@@ -301,18 +250,10 @@
               ></el-input>
             </el-form-item>
             <el-form-item label="年级">
-              <el-input
-                class="input"
-                v-model="examStudentForm.grade"
-                auto-complete="off"
-              ></el-input>
+              <el-input class="input" v-model="examStudentForm.grade" auto-complete="off"></el-input>
             </el-form-item>
             <el-form-item label="采集人">
-              <el-input
-                class="input"
-                v-model="examStudentForm.infoCollector"
-                maxlength="20"
-              ></el-input>
+              <el-input class="input" v-model="examStudentForm.infoCollector" maxlength="20"></el-input>
             </el-form-item>
             <el-form-item label="电话">
               <el-input
@@ -323,11 +264,7 @@
               ></el-input>
             </el-form-item>
             <el-form-item label="试卷袋编码">
-              <el-input
-                class="input"
-                v-model="examStudentForm.ext1"
-                maxlength="20"
-              ></el-input>
+              <el-input class="input" v-model="examStudentForm.ext1" maxlength="20"></el-input>
             </el-form-item>
           </el-form>
           <div style="text-align: center;">
@@ -370,8 +307,7 @@
                   :value="item.id"
                   :key="item.id"
                   :disabled="!item.enable"
-                >
-                </el-option>
+                ></el-option>
               </el-select>
             </el-form-item>
             <el-form-item label="姓名" prop="studentName">
@@ -383,12 +319,7 @@
               ></el-input>
             </el-form-item>
             <el-form-item label="学号" prop="studentCode">
-              <el-input
-                class="input"
-                v-model="examStudentForm.studentCode"
-                maxlength="20"
-                readonly
-              ></el-input>
+              <el-input class="input" v-model="examStudentForm.studentCode" maxlength="20" readonly></el-input>
             </el-form-item>
             <el-form-item label="身份证号" prop="identityNumber">
               <el-input
@@ -417,8 +348,7 @@
                   :value="item.id"
                   :key="item.id"
                   :disabled="!item.enable"
-                >
-                </el-option>
+                ></el-option>
               </el-select>
             </el-form-item>
             <el-form-item label="课程" prop="courseId">
@@ -439,23 +369,12 @@
                   :value="item.id"
                   :key="item.id"
                   :disabled="!item.enable"
-                >
-                </el-option>
+                ></el-option>
               </el-select>
             </el-form-item>
             <el-form-item label="试卷类型" prop="paperType">
-              <el-select
-                class="input"
-                v-model="examStudentForm.paperType"
-                placeholder="请选择"
-              >
-                <el-option
-                  v-for="item in paperTypeList"
-                  :label="item"
-                  :value="item"
-                  :key="item"
-                >
-                </el-option>
+              <el-select class="input" v-model="examStudentForm.paperType" placeholder="请选择">
+                <el-option v-for="item in paperTypeList" :label="item" :value="item" :key="item"></el-option>
               </el-select>
             </el-form-item>
             <el-form-item label="专业">
@@ -483,11 +402,7 @@
               ></el-input>
             </el-form-item>
             <el-form-item label="采集人">
-              <el-input
-                class="input"
-                v-model="examStudentForm.infoCollector"
-                maxlength="20"
-              ></el-input>
+              <el-input class="input" v-model="examStudentForm.infoCollector" maxlength="20"></el-input>
             </el-form-item>
             <el-form-item label="电话">
               <el-input
@@ -498,11 +413,7 @@
               ></el-input>
             </el-form-item>
             <el-form-item label="试卷袋编码">
-              <el-input
-                class="input"
-                v-model="examStudentForm.ext1"
-                maxlength="20"
-              ></el-input>
+              <el-input class="input" v-model="examStudentForm.ext1" maxlength="20"></el-input>
             </el-form-item>
           </el-form>
           <div style="text-align: center;">
@@ -512,11 +423,7 @@
         </el-dialog>
 
         <!-- 复制弹窗 -->
-        <el-dialog
-          title="考生信息复制"
-          width="500px"
-          :visible.sync="studentCopyDialog"
-        >
+        <el-dialog title="考生信息复制" width="500px" :visible.sync="studentCopyDialog">
           <el-form
             :model="studentCopyForm"
             :rules="rules"
@@ -541,8 +448,7 @@
                   :label="item.name"
                   :value="item.id"
                   :key="item.id"
-                >
-                </el-option>
+                ></el-option>
               </el-select>
             </el-form-item>
             <el-form-item label="目标考试批次" prop="targetExamId">
@@ -562,8 +468,7 @@
                   :value="item.id"
                   :key="item.id"
                   :disabled="!item.enable"
-                >
-                </el-option>
+                ></el-option>
               </el-select>
             </el-form-item>
             <div style="text-align: center;">
@@ -582,98 +487,63 @@
           style="width: 100%;text-align:center;"
           @selection-change="selectChange"
         >
-          <el-table-column
-            type="selection"
-            width="40"
-            fixed="left"
-          ></el-table-column>
-          <el-table-column
-            prop="id"
-            label="ID"
-            sortable
-            fixed="left"
-          ></el-table-column>
+          <el-table-column type="selection" width="40" fixed="left"></el-table-column>
+          <el-table-column prop="id" label="ID" sortable fixed="left"></el-table-column>
           <el-table-column label="考生" fixed="left">
             <template slot-scope="scope">
               <el-popover trigger="hover" placement="left">
                 <div style="font-size: 18px;font-family: 新宋体">
                   <tr>
                     <td style="color: green">姓名</td>
-                    <td style="color:purple;padding-left: 20px;">
-                      {{ scope.row.studentName }}
-                    </td>
+                    <td style="color:purple;padding-left: 20px;">{{ scope.row.studentName }}</td>
                   </tr>
                   <tr>
                     <td style="color: green">身份证号</td>
-                    <td style="color:purple;padding-left: 20px;">
-                      {{ scope.row.identityNumber }}
-                    </td>
+                    <td style="color:purple;padding-left: 20px;">{{ scope.row.identityNumber }}</td>
                   </tr>
                   <tr>
                     <td style="color: green">学号</td>
-                    <td style="color:purple;padding-left: 20px;">
-                      {{ scope.row.studentCode }}
-                    </td>
+                    <td style="color:purple;padding-left: 20px;">{{ scope.row.studentCode }}</td>
                   </tr>
                   <tr>
                     <td style="color: green">学习中心名称</td>
-                    <td style="color:purple;padding-left: 20px;">
-                      {{ scope.row.orgName }}
-                    </td>
+                    <td style="color:purple;padding-left: 20px;">{{ scope.row.orgName }}</td>
                   </tr>
                   <tr>
                     <td style="color: green">学习中心编码</td>
-                    <td style="color:purple;padding-left: 20px;">
-                      {{ scope.row.orgCode }}
-                    </td>
+                    <td style="color:purple;padding-left: 20px;">{{ scope.row.orgCode }}</td>
                   </tr>
                   <tr>
                     <td style="color: green">课程名称</td>
-                    <td style="color:purple;padding-left: 20px;">
-                      {{ scope.row.courseName }}
-                    </td>
+                    <td style="color:purple;padding-left: 20px;">{{ scope.row.courseName }}</td>
                   </tr>
                   <tr>
                     <td style="color: green">课程编码</td>
-                    <td style="color:purple;padding-left: 20px;">
-                      {{ scope.row.courseCode }}
-                    </td>
+                    <td style="color:purple;padding-left: 20px;">{{ scope.row.courseCode }}</td>
                   </tr>
                   <tr>
                     <td style="color: green">年级</td>
-                    <td style="color:purple;padding-left: 20px;">
-                      {{ scope.row.grade }}
-                    </td>
+                    <td style="color:purple;padding-left: 20px;">{{ scope.row.grade }}</td>
                   </tr>
                   <tr>
                     <td style="color: green">电话</td>
-                    <td style="color:purple;padding-left: 20px;">
-                      {{ scope.row.phone }}
-                    </td>
+                    <td style="color:purple;padding-left: 20px;">{{ scope.row.phone }}</td>
                   </tr>
                   <tr>
                     <td style="color: green">专业</td>
-                    <td style="color:purple;padding-left: 20px;">
-                      {{ scope.row.specialtyName }}
-                    </td>
+                    <td style="color:purple;padding-left: 20px;">{{ scope.row.specialtyName }}</td>
                   </tr>
                   <tr>
                     <td style="color: green">采集人</td>
-                    <td style="color:purple;padding-left: 20px;">
-                      {{ scope.row.infoCollector }}
-                    </td>
+                    <td style="color:purple;padding-left: 20px;">{{ scope.row.infoCollector }}</td>
                   </tr>
                   <tr>
                     <td style="color: green">考点</td>
-                    <td style="color:purple;padding-left: 20px;">
-                      {{ scope.row.examSite }}
-                    </td>
+                    <td style="color:purple;padding-left: 20px;">{{ scope.row.examSite }}</td>
                   </tr>
                   <tr>
                     <td style="color: green">试卷袋编码</td>
-                    <td style="color:purple;padding-left: 20px;">
-                      {{ scope.row.ext1 }}
-                    </td>
+                    <td style="color:purple;padding-left: 20px;">{{ scope.row.ext1 }}</td>
                   </tr>
                 </div>
 
@@ -683,57 +553,21 @@
               </el-popover>
             </template>
           </el-table-column>
-          <el-table-column prop="studentCode" label="学号" sortable>
-          </el-table-column>
-          <el-table-column
-            prop="identityNumber"
-            label="身份证号"
-            width="120"
-            sortable
-          >
-          </el-table-column>
-          <el-table-column prop="orgName" label="学习中心" width="120" sortable>
-          </el-table-column>
-          <el-table-column
-            prop="examName"
-            label="考试名称"
-            width="120"
-            sortable
-          >
-          </el-table-column>
-          <el-table-column
-            prop="courseName"
-            label="课程名称"
-            width="120"
-            sortable
-          >
-          </el-table-column>
-          <el-table-column
-            prop="updateTime"
-            width="155"
-            label="更新时间"
-            sortable
-          >
-          </el-table-column>
+          <el-table-column prop="studentCode" label="学号" sortable></el-table-column>
+          <el-table-column prop="ext2" label="身份证号" width="120" sortable></el-table-column>
+          <el-table-column prop="orgName" label="学习中心" width="120" sortable></el-table-column>
+          <el-table-column prop="examName" label="考试名称" width="120" sortable></el-table-column>
+          <el-table-column prop="courseName" label="课程名称" width="120" sortable></el-table-column>
+          <el-table-column prop="updateTime" width="155" label="更新时间" sortable></el-table-column>
           <el-table-column width="50" label="状态">
             <span slot-scope="scope">
               <span v-if="scope.row.enable">
-                <el-tooltip
-                  class="item"
-                  effect="dark"
-                  content="启用"
-                  placement="left"
-                >
+                <el-tooltip class="item" effect="dark" content="启用" placement="left">
                   <i class="el-icon-success" style="color:green;"></i>
                 </el-tooltip>
               </span>
               <span v-else>
-                <el-tooltip
-                  class="item"
-                  effect="dark"
-                  content="禁用"
-                  placement="left"
-                >
+                <el-tooltip class="item" effect="dark" content="禁用" placement="left">
                   <i class="el-icon-error" style="color:red;"></i>
                 </el-tooltip>
               </span>
@@ -750,9 +584,7 @@
                   :disabled="!scope.row.photoPath"
                   icon="el-icon-picture"
                   @click="showPhoto(scope.row)"
-                >
-                  查看照片
-                </el-button>
+                >查看照片</el-button>
                 <el-button
                   v-if="rolePrivileges.update_examStudent"
                   size="mini"
@@ -760,9 +592,7 @@
                   plain
                   icon="el-icon-edit"
                   @click="openUpdateDialog(scope.row)"
-                >
-                  编辑
-                </el-button>
+                >编辑</el-button>
 
                 <el-dropdown
                   style="margin-left: 10px;"
@@ -772,7 +602,8 @@
                   "
                 >
                   <el-button type="primary" plain size="mini">
-                    更多<i class="el-icon-arrow-down el-icon--right"></i>
+                    更多
+                    <i class="el-icon-arrow-down el-icon--right"></i>
                   </el-button>
                   <el-dropdown-menu slot="dropdown">
                     <el-dropdown-item>
@@ -782,9 +613,8 @@
                         type="danger"
                         icon="el-icon-delete"
                         @click="deleteExamStudent(scope.row)"
-                        >删除
-                      </el-button></el-dropdown-item
-                    >
+                      >删除</el-button>
+                    </el-dropdown-item>
                     <el-dropdown-item>
                       <el-button
                         v-if="
@@ -797,8 +627,7 @@
                         plain
                         icon="el-icon-check"
                         @click="enableExamStudent(scope.row)"
-                        >启用
-                      </el-button>
+                      >启用</el-button>
                       <el-button
                         v-else-if="
                           rolePrivileges.change_exam_student_availability
@@ -807,8 +636,7 @@
                         type="danger"
                         icon="el-icon-close"
                         @click="disableExamStudent(scope.row)"
-                        >禁用
-                      </el-button>
+                      >禁用</el-button>
                     </el-dropdown-item>
                   </el-dropdown-menu>
                 </el-dropdown>
@@ -825,8 +653,7 @@
             @size-change="handleSizeChange"
             layout="total, sizes, prev, pager, next, jumper"
             :total="total"
-          >
-          </el-pagination>
+          ></el-pagination>
         </div>
 
         <el-dialog title="错误提示" v-model="errDialog">
@@ -834,20 +661,13 @@
             class="text-danger"
             v-for="errMessage in errMessages"
             :key="errMessage.lineNum"
-          >
-            第{{ errMessage.lineNum }}行:{{ errMessage.msg }}
-          </div>
+          >第{{ errMessage.lineNum }}行:{{ errMessage.msg }}</div>
           <span slot="footer" class="dialog-footer">
             <el-button @click="errDialog = false">确定</el-button>
           </span>
         </el-dialog>
 
-        <el-dialog
-          @close="closePhotoDialog"
-          title="照片"
-          :visible.sync="photoDialog"
-          width="300px"
-        >
+        <el-dialog @close="closePhotoDialog" title="照片" :visible.sync="photoDialog" width="300px">
           <img :src="photo.url" height="100%" width="100%" />
         </el-dialog>
       </div>

+ 44 - 5
src/modules/examwork/view/offlineExam.vue

@@ -162,7 +162,7 @@
                       class="input"
                     >
                       <el-radio label="ORG_BASED">机构特殊设置</el-radio>
-                      <el-radio label="STUDENT_BASED">学生特殊设置</el-radio>
+                      <!-- <el-radio label="STUDENT_BASED">学生特殊设置</el-radio> -->
                     </el-radio-group>
                   </el-form-item>
                 </el-row>
@@ -195,10 +195,16 @@
                 </el-row>
                 <el-row>
                   <el-form-item
+                    prop="uploadFileType"
+                    ref="uploadFileType"
                     label="附件类型"
                     :label-width="style.label_width_tab1"
                   >
-                    <el-checkbox-group v-model="uploadFileType" class="input">
+                    <el-checkbox-group
+                      v-model="uploadFileType"
+                      :disabled="this.uploadFileTypeDis()"
+                      class="input"
+                    >
                       <el-checkbox label="ZIP" key="ZIP"></el-checkbox>
                       <el-checkbox label="PDF" key="PDF"></el-checkbox>
                     </el-checkbox-group>
@@ -255,7 +261,16 @@ let validateExamDatetimeRange = (rule, value, callback) => {
     callback();
   }
 };
-
+let validateUploadFileType = (rule, value, callback) => {
+  if (
+    _this.form.properties.CAN_UPLOAD_ATTACHMENT == "true" &&
+    _this.uploadFileType.length == 0
+  ) {
+    callback(new Error("请至少勾选一个附件类型"));
+  } else {
+    callback();
+  }
+};
 export default {
   components: { LinkTitlesCustom },
   data() {
@@ -277,7 +292,7 @@ export default {
         duration: 120,
         enable: "true",
         specialSettingsEnabled: false,
-        specialSettingsType: 'ORG_BASED',
+        specialSettingsType: "ORG_BASED",
         properties: {
           CAN_UPLOAD_ATTACHMENT: "true",
           OFFLINE_UPLOAD_FILE_TYPE: "",
@@ -295,12 +310,25 @@ export default {
             validator: validateExamDatetimeRange,
             trigger: "blur"
           }
+        ],
+        uploadFileType: [
+          {
+            validator: validateUploadFileType,
+            trigger: "change"
+          }
         ]
       }
     };
   },
 
   methods: {
+    uploadFileTypeDis() {
+      if (this.form.properties.CAN_UPLOAD_ATTACHMENT == "true") {
+        return false;
+      } else {
+        return true;
+      }
+    },
     init() {
       if (this.examId != "add") {
         let url = EXAM_WORK_API + "/exam/" + this.examId;
@@ -345,7 +373,7 @@ export default {
       this.form.properties.OFFLINE_UPLOAD_FILE_TYPE = JSON.stringify(
         this.uploadFileType
       );
-      console.log(this.form);
+      console.log(this.uploadFileType);
       let url = EXAM_WORK_API + "/exam";
       this.$refs.form.validate(valid => {
         if (valid) {
@@ -391,6 +419,17 @@ export default {
     _this = this;
     this.examId = this.$route.params.id;
     this.init();
+  },
+  watch: {
+    "form.properties.CAN_UPLOAD_ATTACHMENT": {
+      immediate: false,
+      handler(val) {
+        if (val != "true") {
+          this.uploadFileType = [];
+          this.$refs["uploadFileType"].resetField();
+        }
+      }
+    }
   }
 };
 </script>

+ 1 - 19
src/modules/examwork/view/onlineExam.vue

@@ -196,7 +196,7 @@
                       class="input"
                     >
                       <el-radio label="ORG_BASED">机构特殊设置</el-radio>
-                      <el-radio label="STUDENT_BASED">学生特殊设置</el-radio>
+                      <!-- <el-radio label="STUDENT_BASED">学生特殊设置</el-radio> -->
                     </el-radio-group>
                   </el-form-item>
                 </el-row>
@@ -1049,24 +1049,6 @@ let validateOutFreezeTimeFaceVerifyStartMinute = (rule, value, callback) => {
   let v = _this.form.properties.OUT_FREEZE_TIME_FACE_VERIFY_START_MINUTE;
   let freezeTime = _this.form.properties.FREEZE_TIME;
   let duration = _this.form.duration;
-  console.log(
-    "---isFaceVerify.test---" +
-      isFaceVerify +
-      "----equals---" +
-      (isFaceVerify == "true")
-  );
-  console.log(
-    "---addFaceVerifyOutFreezeTime---" +
-      addFaceVerifyOutFreezeTime +
-      "----equals---" +
-      (addFaceVerifyOutFreezeTime == "true")
-  );
-  console.log(
-    "---faceVerifyScheme---" +
-      faceVerifyScheme +
-      "----equals---" +
-      (faceVerifyScheme == "S2")
-  );
   if (
     isFaceVerify == "true" &&
     addFaceVerifyOutFreezeTime == "true" &&

+ 14 - 24
src/modules/marking/views/TpMain.vue

@@ -76,9 +76,11 @@
 <script>
 import { Drawing } from "../canvas/mark_sign";
 import { EVENTHUB } from "../constants/constants";
+import { MARKING_API } from "@/constants/constants";
 import printJS from "print-js";
 import "viewerjs/dist/viewer.css";
 import Viewer from "viewerjs";
+import { mapState } from "vuex";
 export default {
   data() {
     return {
@@ -297,30 +299,17 @@ export default {
       printJS({ printable: "answer-content", type: "html" });
     },
     pdfDown() {
-      var url = this.studentPaper.studentSubjectiveHtml;
-      if (url != "undefined" && url != "") {
-        url = new URL(url);
-        url = url.pathname;
-        url = decodeURIComponent(url);
-        let xhr = new XMLHttpRequest();
-        xhr.responseType = "blob";
-        xhr.open("GET", url, true);
-        xhr.onload = function() {
-          if (this.status === 200) {
-            let blob = this.response;
-            let reader = new FileReader();
-            reader.readAsDataURL(blob);
-            reader.onload = function(e) {
-              let a = document.createElement("a");
-              a.download = url.substr(url.lastIndexOf("/") + 1);
-              a.href = e.target.result;
-              document.body.appendChild(a);
-              a.click();
-              a.parentNode.removeChild(a);
-            };
-          }
-        };
-        xhr.send();
+      var fileurl = this.studentPaper.studentSubjectiveHtml;
+      if (fileurl != "undefined" && fileurl != "") {
+        let url =
+          MARKING_API +
+          "/fileDownLoad?url=" +
+          fileurl +
+          "&$key=" +
+          this.user.key +
+          "&$token=" +
+          this.user.token;
+        window.location.href = url;
       } else {
         this.$notify({
           message: "离线url为空",
@@ -330,6 +319,7 @@ export default {
     }
   },
   computed: {
+    ...mapState({ user: state => state.user }),
     itemTitle() {
       if (!this.markSign) return "无";
       var title = "无";

+ 0 - 5
src/modules/oe/views/absent.vue

@@ -51,11 +51,6 @@
               label="学习中心"
               prop="orgName"
             ></el-table-column>
-            <el-table-column
-              sortable
-              label="考点"
-              prop="examSiteName"
-            ></el-table-column>
             <el-table-column
               sortable
               label="姓名"

+ 25 - 107
src/modules/oe/views/examScheduling.vue

@@ -48,43 +48,32 @@
           size="small"
           type="primary"
           icon="el-icon-search"
-          >查询</el-button
-        >
+        >查询</el-button>
         <el-button
           size="small"
           type="primary"
           icon="el-icon-more"
           v-if="!showAllCondition"
           @click="showMoreCondition"
-          >高级查询</el-button
-        >
+        >高级查询</el-button>
         <el-button
           size="small"
           type="primary"
           v-if="showAllCondition"
           @click="showSimpleCondition"
-          >简单查询</el-button
-        >
+        >简单查询</el-button>
         <el-button
           size="small"
           icon="el-icon-refresh"
           @click="resetForm"
           class="margin-bottom-10"
-          >重置</el-button
-        >
+        >重置</el-button>
       </el-col>
       <el-row>
-        <el-col
-          :span="24"
-          v-show="currentPagePrivileges.EXAM_PARTICULARS_EXPORT"
-        >
+        <el-col :span="24" v-show="currentPagePrivileges.EXAM_PARTICULARS_EXPORT">
           <div class="block-seperator"></div>
           <span>操作:</span>
-          <commonExportVue
-            :form="form"
-            :exportUrl="exportUrl"
-            :exportFileName="exportFileName"
-          ></commonExportVue>
+          <commonExportVue :form="form" :exportUrl="exportUrl" :exportFileName="exportFileName"></commonExportVue>
         </el-col>
       </el-row>
       <el-row class="margin-top-10">
@@ -97,78 +86,18 @@
             :data="tableData"
             border
           >
-            <el-table-column
-              sortable
-              label="学习中心"
-              prop="orgName"
-              width="120"
-            ></el-table-column>
-            <el-table-column
-              sortable
-              label="姓名"
-              prop="studentName"
-              width="120"
-            ></el-table-column>
-            <el-table-column
-              sortable
-              label="身份证号"
-              prop="identityNumber"
-              width="120"
-            ></el-table-column>
-            <el-table-column
-              sortable
-              label="学号"
-              prop="studentCode"
-              width="120"
-            ></el-table-column>
-            <el-table-column
-              sortable
-              label="课程"
-              prop="courseName"
-              width="120"
-            ></el-table-column>
-            <el-table-column
-              sortable
-              label="课程层次"
-              prop="courseLevel"
-              width="120"
-            ></el-table-column>
-            <el-table-column
-              sortable
-              label="专业"
-              prop="specialtyName"
-              width="120"
-            ></el-table-column>
-            <el-table-column
-              sortable
-              label="已考次数"
-              prop="normalExamTimes"
-              width="120"
-            ></el-table-column>
-            <el-table-column
-              sortable
-              label="学生电话"
-              prop="phone"
-              width="120"
-            ></el-table-column>
-            <el-table-column
-              sortable
-              label="年级"
-              prop="grade"
-              width="120"
-            ></el-table-column>
-            <el-table-column
-              sortable
-              label="采集人"
-              prop="infoCollector"
-              width="120"
-            ></el-table-column>
-            <el-table-column
-              sortable
-              label="完成状态"
-              prop="finishedStatus"
-              width="120"
-            ></el-table-column>
+            <el-table-column sortable label="学习中心" prop="orgName" width="120"></el-table-column>
+            <el-table-column sortable label="姓名" prop="studentName" width="120"></el-table-column>
+            <el-table-column sortable label="身份证号" prop="identityNumber" width="120"></el-table-column>
+            <el-table-column sortable label="学号" prop="studentCode" width="120"></el-table-column>
+            <el-table-column sortable label="课程" prop="courseName" width="120"></el-table-column>
+            <el-table-column sortable label="课程层次" prop="courseLevel" width="120"></el-table-column>
+            <el-table-column sortable label="专业" prop="specialtyName" width="120"></el-table-column>
+            <el-table-column sortable label="已考次数" prop="usedNum" width="120"></el-table-column>
+            <el-table-column sortable label="学生电话" prop="phone" width="120"></el-table-column>
+            <el-table-column sortable label="年级" prop="grade" width="120"></el-table-column>
+            <el-table-column sortable label="采集人" prop="infoCollector" width="120"></el-table-column>
+            <el-table-column sortable label="完成状态" prop="finishedStatus" width="120"></el-table-column>
             <!-- <el-table-column sortable label="完成状态" width="120">
               <template slot-scope="scope">
                 <span> {{ scope.row.finishedStatus }} </span>
@@ -185,8 +114,7 @@
                       icon="el-icon-view"
                       @click="previewPaper(scope.row.examStudentId)"
                       v-if="scope.row.examType == 'OFFLINE'"
-                      >查看考题</el-button
-                    >
+                    >查看考题</el-button>
                   </el-col>
                 </el-row>
                 <el-row class="operateRow">
@@ -198,8 +126,7 @@
                       icon="el-icon-download"
                       @click="exportPaper(scope.row.examStudentId)"
                       v-if="scope.row.examType == 'OFFLINE'"
-                      >下载考题</el-button
-                    >
+                    >下载考题</el-button>
                   </el-col>
                 </el-row>
                 <el-row class="operateRow">
@@ -214,8 +141,7 @@
                         scope.row.examType == 'OFFLINE' &&
                           scope.row.canUploadAttachment
                       "
-                      >上传作答</el-button
-                    >
+                    >上传作答</el-button>
                   </el-col>
                 </el-row>
                 <el-row class="operateRow">
@@ -230,8 +156,7 @@
                         scope.row.examType == 'OFFLINE' &&
                           scope.row.offlineFileUrl
                       "
-                      >下载作答</el-button
-                    >
+                    >下载作答</el-button>
                   </el-col>
                 </el-row>
               </template>
@@ -270,15 +195,8 @@
               <div>温馨提示:仅支持pdf和zip文件,文件大小请不要超过30M!</div>
             </el-form-item>
             <div class="dialog-footer">
-              <el-button @click="uploadAnswerDialogVisible = false"
-                >取 消</el-button
-              >
-              <el-button
-                :disabled="!offlineAnswerFile"
-                type="primary"
-                @click="doUploadAnswer"
-                >确 定</el-button
-              >
+              <el-button @click="uploadAnswerDialogVisible = false">取 消</el-button>
+              <el-button :disabled="!offlineAnswerFile" type="primary" @click="doUploadAnswer">确 定</el-button>
             </div>
           </el-form>
         </el-dialog>
@@ -583,7 +501,7 @@ export default {
           param.append("file", this.offlineAnswerFile);
           param.append("examRecordDataId", this.currentOfflineExamRecordDataId);
           this.$http
-            .post("/api/ecs_oe_student/offlineExam/submitPaper", param, config)
+            .post("/api/ecs_oe_admin/offlineExam/submitPaper", param, config)
             .then(() => {
               this.$notify({
                 title: "提示",

+ 6 - 3
src/modules/questions/component/ckeditor.vue

@@ -56,7 +56,7 @@ export default {
       language: this.language,
       height: this.height,
       width: this.width,
-      extraPlugins: this.extraplugins + ",base64image,pastebase64",
+      extraPlugins: this.extraplugins + ",base64image,pastebase64,image2",
       toolbarGroups: [
         { name: "clipboard", groups: ["clipboard", "undo"] },
         {
@@ -79,9 +79,9 @@ export default {
         { name: "colors", groups: ["colors"] },
         { name: "about", groups: ["about"] }
       ],
-      removePlugins: "bidi,colorbutton",
+      removePlugins: "bidi,colorbutton,image",
       removeButtons:
-        "Font,FontSize,Image,Styles,Format,ShowBlocks,Iframe,PageBreak,Smiley,Flash,Language,JustifyBlock,JustifyRight,JustifyCenter,JustifyLeft,CreateDiv,CopyFormatting,ImageButton,Button,HiddenField,Select,Textarea,TextField,Radio,Checkbox,Form,BGColor,SelectAll,Replace,Find,Templates,Print,Preview,NewPage,Save,Subscript,Superscript,HorizontalRule,Unlink,Link,Scayt,Cut,Copy,Paste,PasteText,PasteFromWord,Maximize,NumberedList,BulletedList,Indent,Outdent,Blockquote,About,RemoveFormat,Strike"
+        "Font,FontSize,Styles,Format,ShowBlocks,Iframe,PageBreak,Smiley,Flash,Language,JustifyBlock,JustifyRight,JustifyCenter,JustifyLeft,CreateDiv,CopyFormatting,ImageButton,Button,HiddenField,Select,Textarea,TextField,Radio,Checkbox,Form,BGColor,SelectAll,Replace,Find,Templates,Print,Preview,NewPage,Save,Subscript,Superscript,HorizontalRule,Unlink,Link,Scayt,Cut,Copy,Paste,PasteText,PasteFromWord,Maximize,NumberedList,BulletedList,Indent,Outdent,Blockquote,About,RemoveFormat,Strike"
     };
     if (this.display !== "inline") {
       window.CKEDITOR.replace(this.id, config);
@@ -109,4 +109,7 @@ export default {
   display: table;
   clear: both;
 }
+.cke_button__image {
+  display: none !important;
+}
 </style>

+ 5 - 0
src/modules/questions/routes/routes.js

@@ -22,6 +22,7 @@ import EditPaper from "../views/EditPaper.vue";
 import PreviewPaper from "../views/PreviewPaper.vue";
 import SelectQuestion from "../views/SelectQuestion.vue";
 import Tips from "../../portal/views/tips/Tips.vue";
+import ExportTemplate from "../views/ExportTemplate.vue";
 
 export default [
   {
@@ -100,6 +101,10 @@ export default [
         path: "export_structure", //导出结构列表
         component: ExportStructure
       },
+      {
+        path: "export_template", //导出模板设定
+        component: ExportTemplate
+      },
       {
         path: "question_list/:isClear", //试题列表
         component: Question

+ 1 - 1
src/modules/questions/views/EditPaper.vue

@@ -5,7 +5,7 @@
     element-loading-text="拼命加载中。。。"
     id="editPaperApp"
   >
-    <ckeditor v-model="examRemark"></ckeditor>
+    <!-- <ckeditor v-model="examRemark"></ckeditor> -->
     <div class="edit-paper-top">
       <div class="edit-paper-top-inline">
         <div class="paper-top-div">

+ 1 - 0
src/modules/questions/views/ExportStructure.vue

@@ -255,6 +255,7 @@
         :model="exportStructure"
         label-position="right"
         label-width="78px"
+        :inline="true"
       >
         <el-row>
           <div style=" color: #f56c6c; margin-bottom: -30px;">*</div>

+ 521 - 0
src/modules/questions/views/ExportTemplate.vue

@@ -0,0 +1,521 @@
+<template>
+  <section class="content">
+    <div class="box-body">
+      <!-- 表单 -->
+      <el-form
+        :inline="true"
+        :model="formSearch"
+        label-width="70px"
+        label-position="right"
+      >
+        <el-row>
+          <el-col :span="6">
+            <el-form-item v-if="isSuperAdmin" label="学校">
+              <el-select
+                v-model="formSearch.rootOrgId"
+                placeholder="请选择"
+                class="search_width"
+                size="small"
+              >
+                <el-option
+                  v-for="item in rootOrgList"
+                  :label="item.name"
+                  :value="item.id"
+                  :key="item.id"
+                />
+              </el-select>
+            </el-form-item>
+          </el-col>
+          <el-col :span="6">
+            <el-form-item label="模板名称">
+              <el-input
+                placeholder="请输入模板名称"
+                class="search_width"
+                v-model="formSearch.fileName"
+                size="small"
+              ></el-input>
+            </el-form-item>
+          </el-col>
+          <el-col :span="6">
+            <el-form-item label="模板类型" prop="type" class="form-item">
+              <el-select
+                v-model="formSearch.type"
+                class="search_width"
+                :clearable="true"
+                size="small"
+              >
+                <el-option label="试卷导出" value="PAPER_EXPORT"> </el-option>
+                <el-option label="答案导出" value="ANWSER_EXPORT"> </el-option>
+                <el-option label="离线试卷" value="OUTLINE_PAPER_EXPORT">
+                </el-option>
+                <el-option label="试卷预览" value="PAPER_VIEW"> </el-option>
+                <el-option label="答案预览" value="ANWSER_VIEW"> </el-option>
+              </el-select>
+            </el-form-item>
+          </el-col>
+          <el-col :span="6">
+            <div class="search_down">
+              <el-button
+                size="small"
+                type="primary"
+                icon="el-icon-search"
+                @click="handleSearchBtn"
+              >
+                查询
+              </el-button>
+              <el-button
+                size="small"
+                type="primary"
+                icon="el-icon-plus"
+                @click="addFile"
+              >
+                新增
+              </el-button>
+            </div>
+          </el-col>
+        </el-row>
+      </el-form>
+
+      <div class="block-seperator"></div>
+      <!-- 页面列表 -->
+      <el-table :data="tableData" border resizable stripe style="width: 100%;">
+        <el-table-column prop="fileName" label="模板名称"> </el-table-column>
+        <el-table-column prop="suffix" width="100" label="文件类型">
+        </el-table-column>
+        <el-table-column width="150" prop="typeName" label="模板类型">
+        </el-table-column>
+        <el-table-column width="150" prop="creationTime" label="上传时间">
+        </el-table-column>
+        <el-table-column width="150" prop="createUser" label="上传人">
+        </el-table-column>
+        <el-table-column label="状态" width="70">
+          <template slot-scope="scope">
+            <span v-if="scope.row.enable">
+              <el-tooltip
+                class="item"
+                effect="dark"
+                content="启用"
+                placement="left"
+              >
+                <i class="el-icon-success" style="color:green;"></i>
+              </el-tooltip>
+            </span>
+            <span v-else>
+              <el-tooltip
+                class="item"
+                effect="dark"
+                content="禁用"
+                placement="left"
+              >
+                <i class="el-icon-error" style="color:red;"></i>
+              </el-tooltip>
+            </span>
+          </template>
+        </el-table-column>
+        <el-table-column :context="_self" label="操作" width="280">
+          <div slot-scope="scope">
+            <el-button
+              size="mini"
+              type="primary"
+              plain
+              @click="downFile(scope.row)"
+              icon="el-icon-download"
+            >
+              下载
+            </el-button>
+            <el-button
+              v-if="!scope.row.enable"
+              size="mini"
+              type="primary"
+              plain
+              @click="enable(scope.row)"
+            >
+              <i class="el-icon-check" aria-hidden="true"></i>启用
+            </el-button>
+            <el-button
+              v-if="scope.row.enable"
+              size="mini"
+              type="danger"
+              @click="disenable(scope.row)"
+            >
+              <i class="el-icon-close" aria-hidden="true"></i>禁用
+            </el-button>
+            <el-button
+              size="mini"
+              type="danger"
+              plain
+              @click="deleteFile(scope.row)"
+              icon="el-icon-remove-outline"
+            >
+              删除
+            </el-button>
+          </div>
+        </el-table-column>
+      </el-table>
+      <div class="page pull-right">
+        <el-pagination
+          v-if="paginationShow"
+          @current-change="handleCurrentChange"
+          :current-page="currentPage"
+          :page-size="pageSize"
+          :page-sizes="[10, 20, 50, 100, 200, 300]"
+          @size-change="handleSizeChange"
+          layout="total, sizes, prev, pager, next, jumper"
+          :total="total"
+        />
+      </div>
+    </div>
+    <el-dialog
+      title="新增模板"
+      width="550px"
+      :visible.sync="fileModel"
+      :close-on-click-modal="false"
+      @close="closeFileModel"
+    >
+      <el-form
+        :inline="true"
+        :model="fileForm"
+        ref="fileForm"
+        :rules="fileRules"
+        label-width="90px"
+        :key="fileModelKey"
+      >
+        <el-row class="form-row">
+          <el-form-item label="模板名称" prop="templateName">
+            <el-input
+              class="pull-length"
+              placeholder="请输入模板名称"
+              v-model="fileForm.templateName"
+            ></el-input>
+          </el-form-item>
+        </el-row>
+        <el-row class="form-row">
+          <el-form-item label="模板类型" prop="type" class="form-item">
+            <el-select v-model="fileForm.type" :clearable="true" class="input">
+              <el-option label="试卷导出" value="PAPER_EXPORT"> </el-option>
+              <el-option label="答案导出" value="ANWSER_EXPORT"> </el-option>
+              <el-option label="离线试卷" value="OUTLINE_PAPER_EXPORT">
+              </el-option>
+              <el-option label="试卷预览" value="PAPER_VIEW"> </el-option>
+              <el-option label="答案预览" value="ANWSER_VIEW"> </el-option>
+            </el-select>
+          </el-form-item>
+        </el-row>
+        <el-row class="form-row">
+          <el-form-item label="文件" prop="dataFile">
+            <el-input
+              class="pull-length"
+              v-model="fileForm.fileName"
+              :readonly="true"
+              placeholder="文件最大限制10M"
+            />
+          </el-form-item>
+          <el-form-item>
+            <el-upload
+              :before-upload="handleUploadFile"
+              accept=".ftl,.zip"
+              action="/upload"
+            >
+              <el-button
+                type="primary"
+                size="small"
+                icon="ios-cloud-upload-outline"
+                >请选择文件
+              </el-button>
+            </el-upload>
+          </el-form-item>
+        </el-row>
+        <el-row class="pull-center">
+          <el-button
+            type="primary"
+            @click="subFile"
+            :loading="this.fileForm.loading"
+            >确定</el-button
+          >
+          <el-button @click="closeFileModel">取消</el-button>
+        </el-row>
+      </el-form>
+    </el-dialog>
+  </section>
+</template>
+<script>
+import { CORE_API, QUESTION_API } from "@/constants/constants.js";
+import { mapState } from "vuex";
+
+export default {
+  name: "ExportTemplate",
+  data() {
+    var reg = /\.zip|\.ftl$/;
+    var validateFile = function(rule, value, callback) {
+      if (value) {
+        var upFileName = value.name;
+        if (value.size > 1024 * 1024 * 10) {
+          return callback(new Error("文件大小限制为10M"));
+        } else if (!upFileName.match(reg)) {
+          return callback(new Error("文件类型只能是ftl或zip"));
+        } else {
+          callback();
+        }
+      } else {
+        callback();
+      }
+    };
+    return {
+      rootOrgList: [],
+      loading: false,
+      fileModel: false,
+      fileModelKey: Math.random(),
+      paginationShow: false,
+      formSearch: {
+        rootOrgId: null,
+        fileName: "",
+        type: ""
+      },
+      fileForm: {
+        templateName: null,
+        type: null,
+        fileName: null,
+        dataFile: null,
+        loading: false
+      },
+      tableData: [],
+      currentPage: 1,
+      pageSize: 10,
+      total: 10,
+      fileRules: {
+        templateName: [
+          {
+            required: true,
+            type: "string",
+            message: "请输入模板名称",
+            trigger: "change"
+          },
+          {
+            pattern: /^[^\\/\\?%#&=\\+]*$/,
+            message: "名称不能包含特殊字符 / ? % # & = +",
+            trigger: "change"
+          }
+        ],
+        type: [
+          {
+            required: true,
+            type: "string",
+            message: "请选择模板类型",
+            trigger: "change"
+          }
+        ],
+        dataFile: [
+          {
+            required: true,
+            type: "object",
+            message: "请选择文件",
+            trigger: "change"
+          },
+          {
+            validator: validateFile,
+            type: "object",
+            trigger: "change"
+          }
+        ]
+      }
+    };
+  },
+  computed: {
+    ...mapState({ user: state => state.user }),
+    isSuperAdmin() {
+      return this.user.roleList.some(role => role.roleCode == "SUPER_ADMIN");
+    }
+  },
+  methods: {
+    downFile(row) {
+      window.location.href = row.fullFilePath;
+    },
+    enable(row) {
+      var url =
+        QUESTION_API +
+        "/exportTemplate/enable/" +
+        this.formSearch.rootOrgId +
+        "/" +
+        row.id;
+      this.$httpWithMsg.put(url).then(() => {
+        this.$notify({
+          type: "success",
+          message: "启用成功!"
+        });
+        this.searchForm();
+      });
+    },
+    disenable(row) {
+      var url =
+        QUESTION_API +
+        "/exportTemplate/disenable/" +
+        this.formSearch.rootOrgId +
+        "/" +
+        row.id;
+      this.$httpWithMsg.put(url).then(() => {
+        this.$notify({
+          type: "success",
+          message: "禁用成功!"
+        });
+        this.searchForm();
+      });
+    },
+    deleteFile(row) {
+      this.$confirm("确定删除?", "提示", {
+        confirmButtonText: "确定",
+        cancelButtonText: "取消",
+        type: "warning"
+      }).then(() => {
+        var url =
+          QUESTION_API +
+          "/exportTemplate/" +
+          this.formSearch.rootOrgId +
+          "/" +
+          row.id;
+        this.$httpWithMsg.delete(url).then(() => {
+          this.$notify({
+            type: "success",
+            message: "删除成功!"
+          });
+          this.searchForm();
+        });
+      });
+    },
+    handleUploadFile(file) {
+      this.fileForm.dataFile = file;
+      this.fileForm.fileName = file.name;
+      return false;
+    },
+    addFile() {
+      if (this.formSearch.rootOrgId == null) {
+        this.$notify({
+          type: "warning",
+          message: "请选择学校"
+        });
+        return;
+      }
+      this.fileModel = true;
+    },
+    async subFile() {
+      const res = await this.$refs.fileForm.validate();
+
+      if (res === false) {
+        return;
+      }
+      this.fileForm.loading = true;
+      var params =
+        "?templateName=" +
+        this.fileForm.templateName +
+        "&type=" +
+        this.fileForm.type;
+      var url =
+        QUESTION_API +
+        "/exportTemplate/add/" +
+        this.formSearch.rootOrgId +
+        params;
+      let formData = new FormData();
+      formData.append("dataFile", this.fileForm.dataFile);
+      this.$httpWithMsg
+        .post(url, formData)
+        .then(() => {
+          this.$notify({
+            type: "success",
+            message: "上传成功!"
+          });
+          this.closeFileModel();
+          this.searchForm();
+        })
+        .finally(() => (this.fileForm.loading = false));
+    },
+    closeFileModel() {
+      this.fileModel = false;
+      this.$refs.fileForm.resetFields();
+      this.fileModelKey = Math.random();
+      this.fileForm.fileName = null;
+      this.fileForm.type = null;
+      this.fileForm.dataFile = null;
+      this.fileForm.templateName = null;
+    },
+    handleSearchBtn() {
+      this.currentPage = 1;
+      this.searchForm();
+    },
+    handleSizeChange(val) {
+      this.pageSize = val;
+      this.currentPage = 1;
+      this.searchForm();
+    },
+    handleCurrentChange(val) {
+      this.currentPage = val;
+      this.searchForm();
+    },
+    //查询
+    searchForm() {
+      if (this.formSearch.rootOrgId == null) {
+        this.$notify({
+          type: "warning",
+          message: "请选择学校"
+        });
+        return;
+      }
+      this.loading = true;
+      var url =
+        QUESTION_API +
+        "/exportTemplate/page/" +
+        this.currentPage +
+        "/" +
+        this.pageSize;
+      this.$httpWithMsg
+        .get(url, { params: this.formSearch })
+        .then(response => {
+          this.tableData = response.data.content;
+          this.total = response.data.totalElements;
+          this.loading = false;
+
+          this.$nextTick(function() {
+            this.paginationShow = true;
+          });
+        })
+        .finally(() => (this.loading = false));
+    },
+    init() {
+      if (this.isSuperAdmin) {
+        this.$httpWithMsg
+          .get(CORE_API + "/org/getRootOrgList")
+          .then(response => {
+            this.rootOrgList = response.data;
+          });
+      } else {
+        this.formSearch.rootOrgId = this.user.rootOrgId;
+        this.searchForm();
+      }
+    }
+  },
+  //初始化查询
+  created() {
+    this.init();
+  }
+};
+</script>
+
+<style scoped src="../styles/Common.css"></style>
+<style scoped>
+.page {
+  margin-top: 10px;
+}
+.pull-length {
+  width: 300px;
+}
+.details-length {
+  width: 400px;
+}
+.pull-center {
+  margin-top: 20px;
+}
+.editForm .el-form-item {
+  margin-bottom: 12px;
+}
+.form-row {
+  margin-top: 20px;
+}
+</style>

Some files were not shown because too many files changed in this diff