Cyr-To-Lat - Version 5.0.0

Version Description

(18.03.2021) = * Introduce tabs on options page * Add options to select post types and statuses for background conversion * Make colors compatible to WP official palette * Fix bug with Polylang when locale is not equal to language slug

Download this release

Release Info

Developer mihdan
Plugin Icon 128x128 Cyr-To-Lat
Version 5.0.0
Comparing to
See all releases

Code changes from version 4.6.4 to 5.0.0

Files changed (37) hide show
  1. assets/css/converter.css +74 -0
  2. assets/css/settings-base.css +25 -0
  3. css/cyr-to-lat-admin.css → assets/css/tables.css +11 -26
  4. {js → assets/js}/acf-field-group.js +0 -0
  5. assets/js/converter/app.js +2 -0
  6. assets/js/tables/app.js +2 -0
  7. classes/class-settings.php +0 -785
  8. cyr-to-lat.php +19 -22
  9. dist/js/settings/app.js +0 -2
  10. languages/cyr2lat-ru_RU.mo +0 -0
  11. languages/cyr2lat-ru_RU.po +158 -176
  12. languages/cyr2lat-sv.po +7 -7
  13. languages/cyr2lat-uk.po +7 -7
  14. readme.txt +13 -3
  15. src/php/Settings/Abstracts/SettingsBase.php +956 -0
  16. src/php/Settings/Abstracts/SettingsInterface.php +23 -0
  17. src/php/Settings/Converter.php +295 -0
  18. src/php/Settings/PluginSettingsBase.php +72 -0
  19. src/php/Settings/Settings.php +165 -0
  20. src/php/Settings/Tables.php +286 -0
  21. {classes → src/php}/background-processes/class-conversion-process.php +0 -0
  22. {classes → src/php}/background-processes/class-post-conversion-process.php +0 -0
  23. {classes → src/php}/background-processes/class-term-conversion-process.php +0 -0
  24. {classes → src/php}/class-acf.php +3 -1
  25. {classes → src/php}/class-admin-notices.php +6 -7
  26. {classes → src/php}/class-conversion-tables.php +0 -0
  27. {classes → src/php}/class-converter.php +21 -19
  28. {classes → src/php}/class-main.php +17 -8
  29. {classes → src/php}/class-requirements.php +18 -13
  30. {classes → src/php}/class-wp-cli.php +0 -0
  31. vendor/autoload.php +1 -1
  32. vendor/composer/ClassLoader.php +4 -2
  33. vendor/composer/InstalledVersions.php +6 -7
  34. vendor/composer/autoload_classmap.php +16 -11
  35. vendor/composer/autoload_real.php +7 -7
  36. vendor/composer/autoload_static.php +20 -15
  37. vendor/composer/installed.php +6 -6
assets/css/converter.css ADDED
@@ -0,0 +1,74 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ #ctl-confirm-popup {
2
+ display: none;
3
+ position: fixed;
4
+ left: 0;
5
+ right: 0;
6
+ top: 0;
7
+ bottom: 0;
8
+ background: rgba(241,241,241,0.8);
9
+ z-index: 9999;
10
+ }
11
+
12
+ #ctl-confirm-content {
13
+ display: block;
14
+ position: fixed;
15
+ left: 50%;
16
+ top: 50%;
17
+ transform: translate(-50%, -50%);
18
+ background: #fff;
19
+ padding: 20px;
20
+ }
21
+
22
+ .helper {
23
+ position: relative;
24
+ float: left;
25
+ left: -1.7em;
26
+ top: 0.4em;
27
+ }
28
+
29
+ .helper-content {
30
+ position: absolute;
31
+ display: none;
32
+ line-height: 1.5em;
33
+ top: 50%;
34
+ left: 50%;
35
+ transform: translate(calc(-50% + 5px), 26px);
36
+ color: #fff;
37
+ font-size: .8em;
38
+ width: 150px;
39
+ background: #333;
40
+ text-align: center;
41
+ border-radius: 3px;
42
+ padding: .618em 1em;
43
+ box-shadow: 0 1px 3px rgba(0, 0, 0, .2);
44
+ }
45
+
46
+ .helper::after {
47
+ font-family: Dashicons;
48
+ font-size: 1.3em;
49
+ speak: none;
50
+ font-weight: 400;
51
+ text-transform: none;
52
+ line-height: 1;
53
+ -webkit-font-smoothing: antialiased;
54
+ text-indent: 0;
55
+ position: absolute;
56
+ top: 0;
57
+ left: 0;
58
+ width: 100%;
59
+ height: 100%;
60
+ text-align: center;
61
+ content: "";
62
+ cursor: help;
63
+ font-variant: normal;
64
+ margin: 0;
65
+ }
66
+
67
+ .helper:hover .helper-content {
68
+ display: block;
69
+ }
70
+
71
+ #ctl-convert-button {
72
+ color: #d63638;
73
+ border-color: #d63638;
74
+ }
assets/css/settings-base.css ADDED
@@ -0,0 +1,25 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ .ctl-settings-tabs {
2
+ line-height: 4.5em;
3
+ background: #fff;
4
+ margin: 10px -20px 0 -20px;
5
+ padding: 0 20px;
6
+ }
7
+
8
+ .ctl-settings-tab {
9
+ display: inline-block;
10
+ font-size: 1.1em;
11
+ text-decoration: none;
12
+ margin-right: 30px;
13
+ color: #646970;
14
+ border-bottom: 2px solid transparent;
15
+ }
16
+
17
+ .ctl-settings-tab:hover {
18
+ color: #666;
19
+ border-color: #646970;
20
+ border-bottom: 2px solid;
21
+ }
22
+
23
+ .ctl-settings-tab.active {
24
+ border-bottom: 2px solid #4f94d4;
25
+ }
css/cyr-to-lat-admin.css → assets/css/tables.css RENAMED
@@ -27,6 +27,7 @@
27
  max-height: 34px;
28
  height: 14px;
29
  line-height: 14px;
 
30
  }
31
 
32
  #ctl-options .nav-tab-wrapper {
@@ -49,7 +50,7 @@ body.js #ctl-options .ctl-table {
49
  clear: both;
50
  width: 100%;
51
  box-sizing: border-box;
52
- border: 1px solid #ccc;
53
  }
54
 
55
  #ctl-options .submit, #ctl-convert-existing-slugs {
@@ -64,12 +65,16 @@ body.js #ctl-options .ctl-table {
64
  }
65
 
66
  #ctl-options .nav-tab-active {
67
- background: #fff;
68
- border-bottom: 1px solid #ccc;
 
69
  }
70
 
71
- .nav-tab-current {
72
- box-shadow: 0 0 5px 5px #fff;
 
 
 
73
  }
74
 
75
  .ctl-table.active {
@@ -134,27 +139,7 @@ body.js #ctl-options .ctl-table {
134
 
135
  #appreciation {
136
  clear: both;
137
- }
138
-
139
- #ctl-confirm-popup {
140
- display: none;
141
- position: fixed;
142
- left: 0;
143
- right: 0;
144
- top: 0;
145
- bottom: 0;
146
- background: rgba(241,241,241,0.8);
147
- z-index: 9999;
148
- }
149
-
150
- #ctl-confirm-content {
151
- display: block;
152
- position: fixed;
153
- left: 50%;
154
- top: 50%;
155
- transform: translate(-50%, -50%);
156
- background: #fff;
157
- padding: 20px;
158
  }
159
 
160
  @media screen and (max-width: 767px) {
27
  max-height: 34px;
28
  height: 14px;
29
  line-height: 14px;
30
+ margin-top: 15px;
31
  }
32
 
33
  #ctl-options .nav-tab-wrapper {
50
  clear: both;
51
  width: 100%;
52
  box-sizing: border-box;
53
+ border: 1px solid #c3c4c7;
54
  }
55
 
56
  #ctl-options .submit, #ctl-convert-existing-slugs {
65
  }
66
 
67
  #ctl-options .nav-tab-active {
68
+ background: #4f94d4;
69
+ color: #fff;
70
+ border-color: #2271b1;
71
  }
72
 
73
+ #ctl-options .nav-tab-current.nav-tab-active {
74
+ /*box-shadow: 0 0 5px 5px #2271b1;*/
75
+ background: #2271b1;
76
+ color: #fff;
77
+ border-color: #0a4b78;
78
  }
79
 
80
  .ctl-table.active {
139
 
140
  #appreciation {
141
  clear: both;
142
+ padding-top: 10px;
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
143
  }
144
 
145
  @media screen and (max-width: 767px) {
{js → assets/js}/acf-field-group.js RENAMED
File without changes
assets/js/converter/app.js ADDED
@@ -0,0 +1,2 @@
 
 
1
+ !function(t){var e={};function r(n){if(e[n])return e[n].exports;var o=e[n]={i:n,l:!1,exports:{}};return t[n].call(o.exports,o,o.exports,r),o.l=!0,o.exports}r.m=t,r.c=e,r.d=function(t,e,n){r.o(t,e)||Object.defineProperty(t,e,{enumerable:!0,get:n})},r.r=function(t){"undefined"!=typeof Symbol&&Symbol.toStringTag&&Object.defineProperty(t,Symbol.toStringTag,{value:"Module"}),Object.defineProperty(t,"__esModule",{value:!0})},r.t=function(t,e){if(1&e&&(t=r(t)),8&e)return t;if(4&e&&"object"==typeof t&&t&&t.__esModule)return t;var n=Object.create(null);if(r.r(n),Object.defineProperty(n,"default",{enumerable:!0,value:t}),2&e&&"string"!=typeof t)for(var o in t)r.d(n,o,function(e){return t[e]}.bind(null,o));return n},r.n=function(t){var e=t&&t.__esModule?function(){return t.default}:function(){return t};return r.d(e,"a",e),e},r.o=function(t,e){return Object.prototype.hasOwnProperty.call(t,e)},r.p="",r(r.s=0)}([function(t,e,r){r(1),t.exports=r(2)},function(t,e){var r="undefined"!=typeof self?self:this,n=function(){function t(){this.fetch=!1,this.DOMException=r.DOMException}return t.prototype=r,new t}();!function(t){!function(e){var r="URLSearchParams"in t,n="Symbol"in t&&"iterator"in Symbol,o="FileReader"in t&&"Blob"in t&&function(){try{return new Blob,!0}catch(t){return!1}}(),i="FormData"in t,s="ArrayBuffer"in t;if(s)var a=["[object Int8Array]","[object Uint8Array]","[object Uint8ClampedArray]","[object Int16Array]","[object Uint16Array]","[object Int32Array]","[object Uint32Array]","[object Float32Array]","[object Float64Array]"],u=ArrayBuffer.isView||function(t){return t&&a.indexOf(Object.prototype.toString.call(t))>-1};function c(t){if("string"!=typeof t&&(t=String(t)),/[^a-z0-9\-#$%&'*+.^_`|~]/i.test(t))throw new TypeError("Invalid character in header field name");return t.toLowerCase()}function f(t){return"string"!=typeof t&&(t=String(t)),t}function h(t){var e={next:function(){var e=t.shift();return{done:void 0===e,value:e}}};return n&&(e[Symbol.iterator]=function(){return e}),e}function d(t){this.map={},t instanceof d?t.forEach((function(t,e){this.append(e,t)}),this):Array.isArray(t)?t.forEach((function(t){this.append(t[0],t[1])}),this):t&&Object.getOwnPropertyNames(t).forEach((function(e){this.append(e,t[e])}),this)}function l(t){if(t.bodyUsed)return Promise.reject(new TypeError("Already read"));t.bodyUsed=!0}function p(t){return new Promise((function(e,r){t.onload=function(){e(t.result)},t.onerror=function(){r(t.error)}}))}function y(t){var e=new FileReader,r=p(e);return e.readAsArrayBuffer(t),r}function b(t){if(t.slice)return t.slice(0);var e=new Uint8Array(t.byteLength);return e.set(new Uint8Array(t)),e.buffer}function m(){return this.bodyUsed=!1,this._initBody=function(t){var e;this._bodyInit=t,t?"string"==typeof t?this._bodyText=t:o&&Blob.prototype.isPrototypeOf(t)?this._bodyBlob=t:i&&FormData.prototype.isPrototypeOf(t)?this._bodyFormData=t:r&&URLSearchParams.prototype.isPrototypeOf(t)?this._bodyText=t.toString():s&&o&&((e=t)&&DataView.prototype.isPrototypeOf(e))?(this._bodyArrayBuffer=b(t.buffer),this._bodyInit=new Blob([this._bodyArrayBuffer])):s&&(ArrayBuffer.prototype.isPrototypeOf(t)||u(t))?this._bodyArrayBuffer=b(t):this._bodyText=t=Object.prototype.toString.call(t):this._bodyText="",this.headers.get("content-type")||("string"==typeof t?this.headers.set("content-type","text/plain;charset=UTF-8"):this._bodyBlob&&this._bodyBlob.type?this.headers.set("content-type",this._bodyBlob.type):r&&URLSearchParams.prototype.isPrototypeOf(t)&&this.headers.set("content-type","application/x-www-form-urlencoded;charset=UTF-8"))},o&&(this.blob=function(){var t=l(this);if(t)return t;if(this._bodyBlob)return Promise.resolve(this._bodyBlob);if(this._bodyArrayBuffer)return Promise.resolve(new Blob([this._bodyArrayBuffer]));if(this._bodyFormData)throw new Error("could not read FormData body as blob");return Promise.resolve(new Blob([this._bodyText]))},this.arrayBuffer=function(){return this._bodyArrayBuffer?l(this)||Promise.resolve(this._bodyArrayBuffer):this.blob().then(y)}),this.text=function(){var t,e,r,n=l(this);if(n)return n;if(this._bodyBlob)return t=this._bodyBlob,e=new FileReader,r=p(e),e.readAsText(t),r;if(this._bodyArrayBuffer)return Promise.resolve(function(t){for(var e=new Uint8Array(t),r=new Array(e.length),n=0;n<e.length;n++)r[n]=String.fromCharCode(e[n]);return r.join("")}(this._bodyArrayBuffer));if(this._bodyFormData)throw new Error("could not read FormData body as text");return Promise.resolve(this._bodyText)},i&&(this.formData=function(){return this.text().then(O)}),this.json=function(){return this.text().then(JSON.parse)},this}d.prototype.append=function(t,e){t=c(t),e=f(e);var r=this.map[t];this.map[t]=r?r+", "+e:e},d.prototype.delete=function(t){delete this.map[c(t)]},d.prototype.get=function(t){return t=c(t),this.has(t)?this.map[t]:null},d.prototype.has=function(t){return this.map.hasOwnProperty(c(t))},d.prototype.set=function(t,e){this.map[c(t)]=f(e)},d.prototype.forEach=function(t,e){for(var r in this.map)this.map.hasOwnProperty(r)&&t.call(e,this.map[r],r,this)},d.prototype.keys=function(){var t=[];return this.forEach((function(e,r){t.push(r)})),h(t)},d.prototype.values=function(){var t=[];return this.forEach((function(e){t.push(e)})),h(t)},d.prototype.entries=function(){var t=[];return this.forEach((function(e,r){t.push([r,e])})),h(t)},n&&(d.prototype[Symbol.iterator]=d.prototype.entries);var E=["DELETE","GET","HEAD","OPTIONS","POST","PUT"];function v(t,e){var r,n,o=(e=e||{}).body;if(t instanceof v){if(t.bodyUsed)throw new TypeError("Already read");this.url=t.url,this.credentials=t.credentials,e.headers||(this.headers=new d(t.headers)),this.method=t.method,this.mode=t.mode,this.signal=t.signal,o||null==t._bodyInit||(o=t._bodyInit,t.bodyUsed=!0)}else this.url=String(t);if(this.credentials=e.credentials||this.credentials||"same-origin",!e.headers&&this.headers||(this.headers=new d(e.headers)),this.method=(r=e.method||this.method||"GET",n=r.toUpperCase(),E.indexOf(n)>-1?n:r),this.mode=e.mode||this.mode||null,this.signal=e.signal||this.signal,this.referrer=null,("GET"===this.method||"HEAD"===this.method)&&o)throw new TypeError("Body not allowed for GET or HEAD requests");this._initBody(o)}function O(t){var e=new FormData;return t.trim().split("&").forEach((function(t){if(t){var r=t.split("="),n=r.shift().replace(/\+/g," "),o=r.join("=").replace(/\+/g," ");e.append(decodeURIComponent(n),decodeURIComponent(o))}})),e}function _(t,e){e||(e={}),this.type="default",this.status=void 0===e.status?200:e.status,this.ok=this.status>=200&&this.status<300,this.statusText="statusText"in e?e.statusText:"OK",this.headers=new d(e.headers),this.url=e.url||"",this._initBody(t)}v.prototype.clone=function(){return new v(this,{body:this._bodyInit})},m.call(v.prototype),m.call(_.prototype),_.prototype.clone=function(){return new _(this._bodyInit,{status:this.status,statusText:this.statusText,headers:new d(this.headers),url:this.url})},_.error=function(){var t=new _(null,{status:0,statusText:""});return t.type="error",t};var w=[301,302,303,307,308];_.redirect=function(t,e){if(-1===w.indexOf(e))throw new RangeError("Invalid status code");return new _(null,{status:e,headers:{location:t}})},e.DOMException=t.DOMException;try{new e.DOMException}catch(t){e.DOMException=function(t,e){this.message=t,this.name=e;var r=Error(t);this.stack=r.stack},e.DOMException.prototype=Object.create(Error.prototype),e.DOMException.prototype.constructor=e.DOMException}function T(t,r){return new Promise((function(n,i){var s=new v(t,r);if(s.signal&&s.signal.aborted)return i(new e.DOMException("Aborted","AbortError"));var a=new XMLHttpRequest;function u(){a.abort()}a.onload=function(){var t,e,r={status:a.status,statusText:a.statusText,headers:(t=a.getAllResponseHeaders()||"",e=new d,t.replace(/\r?\n[\t ]+/g," ").split(/\r?\n/).forEach((function(t){var r=t.split(":"),n=r.shift().trim();if(n){var o=r.join(":").trim();e.append(n,o)}})),e)};r.url="responseURL"in a?a.responseURL:r.headers.get("X-Request-URL");var o="response"in a?a.response:a.responseText;n(new _(o,r))},a.onerror=function(){i(new TypeError("Network request failed"))},a.ontimeout=function(){i(new TypeError("Network request failed"))},a.onabort=function(){i(new e.DOMException("Aborted","AbortError"))},a.open(s.method,s.url,!0),"include"===s.credentials?a.withCredentials=!0:"omit"===s.credentials&&(a.withCredentials=!1),"responseType"in a&&o&&(a.responseType="blob"),s.headers.forEach((function(t,e){a.setRequestHeader(e,t)})),s.signal&&(s.signal.addEventListener("abort",u),a.onreadystatechange=function(){4===a.readyState&&s.signal.removeEventListener("abort",u)}),a.send(void 0===s._bodyInit?null:s._bodyInit)}))}T.polyfill=!0,t.fetch||(t.fetch=T,t.Headers=d,t.Request=v,t.Response=_),e.Headers=d,e.Request=v,e.Response=_,e.fetch=T,Object.defineProperty(e,"__esModule",{value:!0})}({})}(n),n.fetch.ponyfill=!0,delete n.fetch.polyfill;var o=n;(e=o.fetch).default=o.fetch,e.fetch=o.fetch,e.Headers=o.Headers,e.Request=o.Request,e.Response=o.Response,t.exports=e},function(t,e,r){"use strict";var n,o=r(3),i=(n=o)&&n.__esModule?n:{default:n};document.addEventListener("DOMContentLoaded",(function(){new i.default}))},function(t,e,r){"use strict";Object.defineProperty(e,"__esModule",{value:!0});var n=function(){function t(t,e){for(var r=0;r<e.length;r++){var n=e[r];n.enumerable=n.enumerable||!1,n.configurable=!0,"value"in n&&(n.writable=!0),Object.defineProperty(t,n.key,n)}}return function(e,r,n){return r&&t(e.prototype,r),n&&t(e,n),e}}();var o=function(){function t(){!function(t,e){if(!(t instanceof e))throw new TypeError("Cannot call a class as a function")}(this,t),this.CONVERT_FORM_SELECTOR="#ctl-convert-existing-slugs",this.CONVERT_BUTTON_SELECTOR="#ctl-convert-button",this.CONFIRM_POPUP_SELECTOR="#ctl-confirm-popup",this.CONFIRM_OK_SELECTOR="#ctl-confirm-ok",this.CONFIRM_CANCEL_SELECTOR="#ctl-confirm-cancel",this.confirmPopup=document.querySelector(this.CONFIRM_POPUP_SELECTOR),this.bindEvents()}return n(t,[{key:"bindEvents",value:function(){var t=this;document.querySelector(this.CONVERT_BUTTON_SELECTOR).onclick=function(e){return e.preventDefault(),t.confirmPopup.style.display="block",!1},this.confirmPopup.onclick=function(){t.hideConfirmPopup()},document.querySelector(this.CONFIRM_OK_SELECTOR).onclick=function(e){e.stopPropagation(),t.hideConfirmPopup(),document.querySelector(t.CONVERT_FORM_SELECTOR).submit()},document.querySelector(this.CONFIRM_CANCEL_SELECTOR).onclick=function(e){e.stopPropagation(),t.hideConfirmPopup()}}},{key:"hideConfirmPopup",value:function(){this.confirmPopup.style.display="none"}}]),t}();e.default=o}]);
2
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["webpack:///webpack/bootstrap","webpack:///./node_modules/cross-fetch/dist/browser-ponyfill.js","webpack:///./src/js/converter/app.js","webpack:///./src/js/converter/converter.js"],"names":["installedModules","__webpack_require__","moduleId","exports","module","i","l","modules","call","m","c","d","name","getter","o","Object","defineProperty","enumerable","get","r","Symbol","toStringTag","value","t","mode","__esModule","ns","create","key","bind","n","object","property","prototype","hasOwnProperty","p","s","global","self","this","__self__","F","fetch","DOMException","support","Blob","e","viewClasses","isArrayBufferView","ArrayBuffer","isView","obj","indexOf","toString","normalizeName","String","test","TypeError","toLowerCase","normalizeValue","iteratorFor","items","iterator","next","shift","done","undefined","Headers","headers","map","forEach","append","Array","isArray","header","getOwnPropertyNames","consumed","body","bodyUsed","Promise","reject","fileReaderReady","reader","resolve","onload","result","onerror","error","readBlobAsArrayBuffer","blob","FileReader","promise","readAsArrayBuffer","bufferClone","buf","slice","view","Uint8Array","byteLength","set","buffer","Body","_initBody","_bodyInit","_bodyText","isPrototypeOf","_bodyBlob","FormData","_bodyFormData","URLSearchParams","DataView","_bodyArrayBuffer","type","rejected","Error","arrayBuffer","then","text","readAsText","chars","length","fromCharCode","join","readArrayBufferAsText","formData","decode","json","JSON","parse","oldValue","has","callback","thisArg","keys","push","values","entries","methods","Request","input","options","method","upcased","url","credentials","signal","toUpperCase","referrer","form","trim","split","bytes","replace","decodeURIComponent","Response","bodyInit","status","ok","statusText","clone","response","redirectStatuses","redirect","RangeError","location","err","message","stack","constructor","init","request","aborted","xhr","XMLHttpRequest","abortXhr","abort","rawHeaders","getAllResponseHeaders","line","parts","responseURL","responseText","ontimeout","onabort","open","withCredentials","responseType","setRequestHeader","addEventListener","onreadystatechange","readyState","removeEventListener","send","polyfill","ponyfill","ctx","default","document","Converter","CONVERT_FORM_SELECTOR","CONVERT_BUTTON_SELECTOR","CONFIRM_POPUP_SELECTOR","CONFIRM_OK_SELECTOR","CONFIRM_CANCEL_SELECTOR","confirmPopup","querySelector","bindEvents","onclick","event","preventDefault","style","display","hideConfirmPopup","stopPropagation","submit"],"mappings":"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","file":"converter/app.js","sourcesContent":[" \t// The module cache\n \tvar installedModules = {};\n\n \t// The require function\n \tfunction __webpack_require__(moduleId) {\n\n \t\t// Check if module is in cache\n \t\tif(installedModules[moduleId]) {\n \t\t\treturn installedModules[moduleId].exports;\n \t\t}\n \t\t// Create a new module (and put it into the cache)\n \t\tvar module = installedModules[moduleId] = {\n \t\t\ti: moduleId,\n \t\t\tl: false,\n \t\t\texports: {}\n \t\t};\n\n \t\t// Execute the module function\n \t\tmodules[moduleId].call(module.exports, module, module.exports, __webpack_require__);\n\n \t\t// Flag the module as loaded\n \t\tmodule.l = true;\n\n \t\t// Return the exports of the module\n \t\treturn module.exports;\n \t}\n\n\n \t// expose the modules object (__webpack_modules__)\n \t__webpack_require__.m = modules;\n\n \t// expose the module cache\n \t__webpack_require__.c = installedModules;\n\n \t// define getter function for harmony exports\n \t__webpack_require__.d = function(exports, name, getter) {\n \t\tif(!__webpack_require__.o(exports, name)) {\n \t\t\tObject.defineProperty(exports, name, { enumerable: true, get: getter });\n \t\t}\n \t};\n\n \t// define __esModule on exports\n \t__webpack_require__.r = function(exports) {\n \t\tif(typeof Symbol !== 'undefined' && Symbol.toStringTag) {\n \t\t\tObject.defineProperty(exports, Symbol.toStringTag, { value: 'Module' });\n \t\t}\n \t\tObject.defineProperty(exports, '__esModule', { value: true });\n \t};\n\n \t// create a fake namespace object\n \t// mode & 1: value is a module id, require it\n \t// mode & 2: merge all properties of value into the ns\n \t// mode & 4: return value when already ns object\n \t// mode & 8|1: behave like require\n \t__webpack_require__.t = function(value, mode) {\n \t\tif(mode & 1) value = __webpack_require__(value);\n \t\tif(mode & 8) return value;\n \t\tif((mode & 4) && typeof value === 'object' && value && value.__esModule) return value;\n \t\tvar ns = Object.create(null);\n \t\t__webpack_require__.r(ns);\n \t\tObject.defineProperty(ns, 'default', { enumerable: true, value: value });\n \t\tif(mode & 2 && typeof value != 'string') for(var key in value) __webpack_require__.d(ns, key, function(key) { return value[key]; }.bind(null, key));\n \t\treturn ns;\n \t};\n\n \t// getDefaultExport function for compatibility with non-harmony modules\n \t__webpack_require__.n = function(module) {\n \t\tvar getter = module && module.__esModule ?\n \t\t\tfunction getDefault() { return module['default']; } :\n \t\t\tfunction getModuleExports() { return module; };\n \t\t__webpack_require__.d(getter, 'a', getter);\n \t\treturn getter;\n \t};\n\n \t// Object.prototype.hasOwnProperty.call\n \t__webpack_require__.o = function(object, property) { return Object.prototype.hasOwnProperty.call(object, property); };\n\n \t// __webpack_public_path__\n \t__webpack_require__.p = \"\";\n\n\n \t// Load entry module and return exports\n \treturn __webpack_require__(__webpack_require__.s = 0);\n","var global = typeof self !== 'undefined' ? self : this;\nvar __self__ = (function () {\nfunction F() {\nthis.fetch = false;\nthis.DOMException = global.DOMException\n}\nF.prototype = global;\nreturn new F();\n})();\n(function(self) {\n\nvar irrelevant = (function (exports) {\n\n  var support = {\n    searchParams: 'URLSearchParams' in self,\n    iterable: 'Symbol' in self && 'iterator' in Symbol,\n    blob:\n      'FileReader' in self &&\n      'Blob' in self &&\n      (function() {\n        try {\n          new Blob();\n          return true\n        } catch (e) {\n          return false\n        }\n      })(),\n    formData: 'FormData' in self,\n    arrayBuffer: 'ArrayBuffer' in self\n  };\n\n  function isDataView(obj) {\n    return obj && DataView.prototype.isPrototypeOf(obj)\n  }\n\n  if (support.arrayBuffer) {\n    var viewClasses = [\n      '[object Int8Array]',\n      '[object Uint8Array]',\n      '[object Uint8ClampedArray]',\n      '[object Int16Array]',\n      '[object Uint16Array]',\n      '[object Int32Array]',\n      '[object Uint32Array]',\n      '[object Float32Array]',\n      '[object Float64Array]'\n    ];\n\n    var isArrayBufferView =\n      ArrayBuffer.isView ||\n      function(obj) {\n        return obj && viewClasses.indexOf(Object.prototype.toString.call(obj)) > -1\n      };\n  }\n\n  function normalizeName(name) {\n    if (typeof name !== 'string') {\n      name = String(name);\n    }\n    if (/[^a-z0-9\\-#$%&'*+.^_`|~]/i.test(name)) {\n      throw new TypeError('Invalid character in header field name')\n    }\n    return name.toLowerCase()\n  }\n\n  function normalizeValue(value) {\n    if (typeof value !== 'string') {\n      value = String(value);\n    }\n    return value\n  }\n\n  // Build a destructive iterator for the value list\n  function iteratorFor(items) {\n    var iterator = {\n      next: function() {\n        var value = items.shift();\n        return {done: value === undefined, value: value}\n      }\n    };\n\n    if (support.iterable) {\n      iterator[Symbol.iterator] = function() {\n        return iterator\n      };\n    }\n\n    return iterator\n  }\n\n  function Headers(headers) {\n    this.map = {};\n\n    if (headers instanceof Headers) {\n      headers.forEach(function(value, name) {\n        this.append(name, value);\n      }, this);\n    } else if (Array.isArray(headers)) {\n      headers.forEach(function(header) {\n        this.append(header[0], header[1]);\n      }, this);\n    } else if (headers) {\n      Object.getOwnPropertyNames(headers).forEach(function(name) {\n        this.append(name, headers[name]);\n      }, this);\n    }\n  }\n\n  Headers.prototype.append = function(name, value) {\n    name = normalizeName(name);\n    value = normalizeValue(value);\n    var oldValue = this.map[name];\n    this.map[name] = oldValue ? oldValue + ', ' + value : value;\n  };\n\n  Headers.prototype['delete'] = function(name) {\n    delete this.map[normalizeName(name)];\n  };\n\n  Headers.prototype.get = function(name) {\n    name = normalizeName(name);\n    return this.has(name) ? this.map[name] : null\n  };\n\n  Headers.prototype.has = function(name) {\n    return this.map.hasOwnProperty(normalizeName(name))\n  };\n\n  Headers.prototype.set = function(name, value) {\n    this.map[normalizeName(name)] = normalizeValue(value);\n  };\n\n  Headers.prototype.forEach = function(callback, thisArg) {\n    for (var name in this.map) {\n      if (this.map.hasOwnProperty(name)) {\n        callback.call(thisArg, this.map[name], name, this);\n      }\n    }\n  };\n\n  Headers.prototype.keys = function() {\n    var items = [];\n    this.forEach(function(value, name) {\n      items.push(name);\n    });\n    return iteratorFor(items)\n  };\n\n  Headers.prototype.values = function() {\n    var items = [];\n    this.forEach(function(value) {\n      items.push(value);\n    });\n    return iteratorFor(items)\n  };\n\n  Headers.prototype.entries = function() {\n    var items = [];\n    this.forEach(function(value, name) {\n      items.push([name, value]);\n    });\n    return iteratorFor(items)\n  };\n\n  if (support.iterable) {\n    Headers.prototype[Symbol.iterator] = Headers.prototype.entries;\n  }\n\n  function consumed(body) {\n    if (body.bodyUsed) {\n      return Promise.reject(new TypeError('Already read'))\n    }\n    body.bodyUsed = true;\n  }\n\n  function fileReaderReady(reader) {\n    return new Promise(function(resolve, reject) {\n      reader.onload = function() {\n        resolve(reader.result);\n      };\n      reader.onerror = function() {\n        reject(reader.error);\n      };\n    })\n  }\n\n  function readBlobAsArrayBuffer(blob) {\n    var reader = new FileReader();\n    var promise = fileReaderReady(reader);\n    reader.readAsArrayBuffer(blob);\n    return promise\n  }\n\n  function readBlobAsText(blob) {\n    var reader = new FileReader();\n    var promise = fileReaderReady(reader);\n    reader.readAsText(blob);\n    return promise\n  }\n\n  function readArrayBufferAsText(buf) {\n    var view = new Uint8Array(buf);\n    var chars = new Array(view.length);\n\n    for (var i = 0; i < view.length; i++) {\n      chars[i] = String.fromCharCode(view[i]);\n    }\n    return chars.join('')\n  }\n\n  function bufferClone(buf) {\n    if (buf.slice) {\n      return buf.slice(0)\n    } else {\n      var view = new Uint8Array(buf.byteLength);\n      view.set(new Uint8Array(buf));\n      return view.buffer\n    }\n  }\n\n  function Body() {\n    this.bodyUsed = false;\n\n    this._initBody = function(body) {\n      this._bodyInit = body;\n      if (!body) {\n        this._bodyText = '';\n      } else if (typeof body === 'string') {\n        this._bodyText = body;\n      } else if (support.blob && Blob.prototype.isPrototypeOf(body)) {\n        this._bodyBlob = body;\n      } else if (support.formData && FormData.prototype.isPrototypeOf(body)) {\n        this._bodyFormData = body;\n      } else if (support.searchParams && URLSearchParams.prototype.isPrototypeOf(body)) {\n        this._bodyText = body.toString();\n      } else if (support.arrayBuffer && support.blob && isDataView(body)) {\n        this._bodyArrayBuffer = bufferClone(body.buffer);\n        // IE 10-11 can't handle a DataView body.\n        this._bodyInit = new Blob([this._bodyArrayBuffer]);\n      } else if (support.arrayBuffer && (ArrayBuffer.prototype.isPrototypeOf(body) || isArrayBufferView(body))) {\n        this._bodyArrayBuffer = bufferClone(body);\n      } else {\n        this._bodyText = body = Object.prototype.toString.call(body);\n      }\n\n      if (!this.headers.get('content-type')) {\n        if (typeof body === 'string') {\n          this.headers.set('content-type', 'text/plain;charset=UTF-8');\n        } else if (this._bodyBlob && this._bodyBlob.type) {\n          this.headers.set('content-type', this._bodyBlob.type);\n        } else if (support.searchParams && URLSearchParams.prototype.isPrototypeOf(body)) {\n          this.headers.set('content-type', 'application/x-www-form-urlencoded;charset=UTF-8');\n        }\n      }\n    };\n\n    if (support.blob) {\n      this.blob = function() {\n        var rejected = consumed(this);\n        if (rejected) {\n          return rejected\n        }\n\n        if (this._bodyBlob) {\n          return Promise.resolve(this._bodyBlob)\n        } else if (this._bodyArrayBuffer) {\n          return Promise.resolve(new Blob([this._bodyArrayBuffer]))\n        } else if (this._bodyFormData) {\n          throw new Error('could not read FormData body as blob')\n        } else {\n          return Promise.resolve(new Blob([this._bodyText]))\n        }\n      };\n\n      this.arrayBuffer = function() {\n        if (this._bodyArrayBuffer) {\n          return consumed(this) || Promise.resolve(this._bodyArrayBuffer)\n        } else {\n          return this.blob().then(readBlobAsArrayBuffer)\n        }\n      };\n    }\n\n    this.text = function() {\n      var rejected = consumed(this);\n      if (rejected) {\n        return rejected\n      }\n\n      if (this._bodyBlob) {\n        return readBlobAsText(this._bodyBlob)\n      } else if (this._bodyArrayBuffer) {\n        return Promise.resolve(readArrayBufferAsText(this._bodyArrayBuffer))\n      } else if (this._bodyFormData) {\n        throw new Error('could not read FormData body as text')\n      } else {\n        return Promise.resolve(this._bodyText)\n      }\n    };\n\n    if (support.formData) {\n      this.formData = function() {\n        return this.text().then(decode)\n      };\n    }\n\n    this.json = function() {\n      return this.text().then(JSON.parse)\n    };\n\n    return this\n  }\n\n  // HTTP methods whose capitalization should be normalized\n  var methods = ['DELETE', 'GET', 'HEAD', 'OPTIONS', 'POST', 'PUT'];\n\n  function normalizeMethod(method) {\n    var upcased = method.toUpperCase();\n    return methods.indexOf(upcased) > -1 ? upcased : method\n  }\n\n  function Request(input, options) {\n    options = options || {};\n    var body = options.body;\n\n    if (input instanceof Request) {\n      if (input.bodyUsed) {\n        throw new TypeError('Already read')\n      }\n      this.url = input.url;\n      this.credentials = input.credentials;\n      if (!options.headers) {\n        this.headers = new Headers(input.headers);\n      }\n      this.method = input.method;\n      this.mode = input.mode;\n      this.signal = input.signal;\n      if (!body && input._bodyInit != null) {\n        body = input._bodyInit;\n        input.bodyUsed = true;\n      }\n    } else {\n      this.url = String(input);\n    }\n\n    this.credentials = options.credentials || this.credentials || 'same-origin';\n    if (options.headers || !this.headers) {\n      this.headers = new Headers(options.headers);\n    }\n    this.method = normalizeMethod(options.method || this.method || 'GET');\n    this.mode = options.mode || this.mode || null;\n    this.signal = options.signal || this.signal;\n    this.referrer = null;\n\n    if ((this.method === 'GET' || this.method === 'HEAD') && body) {\n      throw new TypeError('Body not allowed for GET or HEAD requests')\n    }\n    this._initBody(body);\n  }\n\n  Request.prototype.clone = function() {\n    return new Request(this, {body: this._bodyInit})\n  };\n\n  function decode(body) {\n    var form = new FormData();\n    body\n      .trim()\n      .split('&')\n      .forEach(function(bytes) {\n        if (bytes) {\n          var split = bytes.split('=');\n          var name = split.shift().replace(/\\+/g, ' ');\n          var value = split.join('=').replace(/\\+/g, ' ');\n          form.append(decodeURIComponent(name), decodeURIComponent(value));\n        }\n      });\n    return form\n  }\n\n  function parseHeaders(rawHeaders) {\n    var headers = new Headers();\n    // Replace instances of \\r\\n and \\n followed by at least one space or horizontal tab with a space\n    // https://tools.ietf.org/html/rfc7230#section-3.2\n    var preProcessedHeaders = rawHeaders.replace(/\\r?\\n[\\t ]+/g, ' ');\n    preProcessedHeaders.split(/\\r?\\n/).forEach(function(line) {\n      var parts = line.split(':');\n      var key = parts.shift().trim();\n      if (key) {\n        var value = parts.join(':').trim();\n        headers.append(key, value);\n      }\n    });\n    return headers\n  }\n\n  Body.call(Request.prototype);\n\n  function Response(bodyInit, options) {\n    if (!options) {\n      options = {};\n    }\n\n    this.type = 'default';\n    this.status = options.status === undefined ? 200 : options.status;\n    this.ok = this.status >= 200 && this.status < 300;\n    this.statusText = 'statusText' in options ? options.statusText : 'OK';\n    this.headers = new Headers(options.headers);\n    this.url = options.url || '';\n    this._initBody(bodyInit);\n  }\n\n  Body.call(Response.prototype);\n\n  Response.prototype.clone = function() {\n    return new Response(this._bodyInit, {\n      status: this.status,\n      statusText: this.statusText,\n      headers: new Headers(this.headers),\n      url: this.url\n    })\n  };\n\n  Response.error = function() {\n    var response = new Response(null, {status: 0, statusText: ''});\n    response.type = 'error';\n    return response\n  };\n\n  var redirectStatuses = [301, 302, 303, 307, 308];\n\n  Response.redirect = function(url, status) {\n    if (redirectStatuses.indexOf(status) === -1) {\n      throw new RangeError('Invalid status code')\n    }\n\n    return new Response(null, {status: status, headers: {location: url}})\n  };\n\n  exports.DOMException = self.DOMException;\n  try {\n    new exports.DOMException();\n  } catch (err) {\n    exports.DOMException = function(message, name) {\n      this.message = message;\n      this.name = name;\n      var error = Error(message);\n      this.stack = error.stack;\n    };\n    exports.DOMException.prototype = Object.create(Error.prototype);\n    exports.DOMException.prototype.constructor = exports.DOMException;\n  }\n\n  function fetch(input, init) {\n    return new Promise(function(resolve, reject) {\n      var request = new Request(input, init);\n\n      if (request.signal && request.signal.aborted) {\n        return reject(new exports.DOMException('Aborted', 'AbortError'))\n      }\n\n      var xhr = new XMLHttpRequest();\n\n      function abortXhr() {\n        xhr.abort();\n      }\n\n      xhr.onload = function() {\n        var options = {\n          status: xhr.status,\n          statusText: xhr.statusText,\n          headers: parseHeaders(xhr.getAllResponseHeaders() || '')\n        };\n        options.url = 'responseURL' in xhr ? xhr.responseURL : options.headers.get('X-Request-URL');\n        var body = 'response' in xhr ? xhr.response : xhr.responseText;\n        resolve(new Response(body, options));\n      };\n\n      xhr.onerror = function() {\n        reject(new TypeError('Network request failed'));\n      };\n\n      xhr.ontimeout = function() {\n        reject(new TypeError('Network request failed'));\n      };\n\n      xhr.onabort = function() {\n        reject(new exports.DOMException('Aborted', 'AbortError'));\n      };\n\n      xhr.open(request.method, request.url, true);\n\n      if (request.credentials === 'include') {\n        xhr.withCredentials = true;\n      } else if (request.credentials === 'omit') {\n        xhr.withCredentials = false;\n      }\n\n      if ('responseType' in xhr && support.blob) {\n        xhr.responseType = 'blob';\n      }\n\n      request.headers.forEach(function(value, name) {\n        xhr.setRequestHeader(name, value);\n      });\n\n      if (request.signal) {\n        request.signal.addEventListener('abort', abortXhr);\n\n        xhr.onreadystatechange = function() {\n          // DONE (success or failure)\n          if (xhr.readyState === 4) {\n            request.signal.removeEventListener('abort', abortXhr);\n          }\n        };\n      }\n\n      xhr.send(typeof request._bodyInit === 'undefined' ? null : request._bodyInit);\n    })\n  }\n\n  fetch.polyfill = true;\n\n  if (!self.fetch) {\n    self.fetch = fetch;\n    self.Headers = Headers;\n    self.Request = Request;\n    self.Response = Response;\n  }\n\n  exports.Headers = Headers;\n  exports.Request = Request;\n  exports.Response = Response;\n  exports.fetch = fetch;\n\n  Object.defineProperty(exports, '__esModule', { value: true });\n\n  return exports;\n\n}({}));\n})(__self__);\n__self__.fetch.ponyfill = true;\n// Remove \"polyfill\" property added by whatwg-fetch\ndelete __self__.fetch.polyfill;\n// Choose between native implementation (global) or custom implementation (__self__)\n// var ctx = global.fetch ? global : __self__;\nvar ctx = __self__; // this line disable service worker support temporarily\nexports = ctx.fetch // To enable: import fetch from 'cross-fetch'\nexports.default = ctx.fetch // For TypeScript consumers without esModuleInterop.\nexports.fetch = ctx.fetch // To enable: import {fetch} from 'cross-fetch'\nexports.Headers = ctx.Headers\nexports.Request = ctx.Request\nexports.Response = ctx.Response\nmodule.exports = exports\n","/**\n * @file Converter Application.\n */\nimport Converter from './converter';\n\ndocument.addEventListener( 'DOMContentLoaded', () => {\n\tnew Converter();\n} );\n","/**\n * @file class Converter.\n */\n\nclass Converter {\n\t/**\n\t * Class constructor.\n\t */\n\tconstructor() {\n\t\tthis.CONVERT_FORM_SELECTOR = '#ctl-convert-existing-slugs';\n\t\tthis.CONVERT_BUTTON_SELECTOR = '#ctl-convert-button';\n\t\tthis.CONFIRM_POPUP_SELECTOR = '#ctl-confirm-popup';\n\t\tthis.CONFIRM_OK_SELECTOR = '#ctl-confirm-ok';\n\t\tthis.CONFIRM_CANCEL_SELECTOR = '#ctl-confirm-cancel';\n\n\t\tthis.confirmPopup = document.querySelector(\n\t\t\tthis.CONFIRM_POPUP_SELECTOR\n\t\t);\n\n\t\tthis.bindEvents();\n\t}\n\n\t/**\n\t * Bind events to methods.\n\t */\n\tbindEvents() {\n\t\tdocument.querySelector( this.CONVERT_BUTTON_SELECTOR ).onclick = (\n\t\t\tevent\n\t\t) => {\n\t\t\tevent.preventDefault();\n\t\t\tthis.confirmPopup.style.display = 'block';\n\t\t\treturn false;\n\t\t};\n\n\t\tthis.confirmPopup.onclick = () => {\n\t\t\tthis.hideConfirmPopup();\n\t\t};\n\n\t\tdocument.querySelector( this.CONFIRM_OK_SELECTOR ).onclick = (\n\t\t\tevent\n\t\t) => {\n\t\t\tevent.stopPropagation();\n\t\t\tthis.hideConfirmPopup();\n\t\t\tdocument.querySelector( this.CONVERT_FORM_SELECTOR ).submit();\n\t\t};\n\n\t\tdocument.querySelector( this.CONFIRM_CANCEL_SELECTOR ).onclick = (\n\t\t\tevent\n\t\t) => {\n\t\t\tevent.stopPropagation();\n\t\t\tthis.hideConfirmPopup();\n\t\t};\n\t}\n\n\t/**\n\t * Hide confirmation popup.\n\t */\n\thideConfirmPopup() {\n\t\tthis.confirmPopup.style.display = 'none';\n\t}\n}\n\nexport default Converter;\n"],"sourceRoot":""}
assets/js/tables/app.js ADDED
@@ -0,0 +1,2 @@
 
 
1
+ !function(e){var t={};function n(r){if(t[r])return t[r].exports;var i=t[r]={i:r,l:!1,exports:{}};return e[r].call(i.exports,i,i.exports,n),i.l=!0,i.exports}n.m=e,n.c=t,n.d=function(e,t,r){n.o(e,t)||Object.defineProperty(e,t,{enumerable:!0,get:r})},n.r=function(e){"undefined"!=typeof Symbol&&Symbol.toStringTag&&Object.defineProperty(e,Symbol.toStringTag,{value:"Module"}),Object.defineProperty(e,"__esModule",{value:!0})},n.t=function(e,t){if(1&t&&(e=n(e)),8&t)return e;if(4&t&&"object"==typeof e&&e&&e.__esModule)return e;var r=Object.create(null);if(n.r(r),Object.defineProperty(r,"default",{enumerable:!0,value:e}),2&t&&"string"!=typeof e)for(var i in e)n.d(r,i,function(t){return e[t]}.bind(null,i));return r},n.n=function(e){var t=e&&e.__esModule?function(){return e.default}:function(){return e};return n.d(t,"a",t),t},n.o=function(e,t){return Object.prototype.hasOwnProperty.call(e,t)},n.p="",n(n.s=0)}([function(e,t,n){n(1),e.exports=n(2)},function(e,t){var n="undefined"!=typeof self?self:this,r=function(){function e(){this.fetch=!1,this.DOMException=n.DOMException}return e.prototype=n,new e}();!function(e){!function(t){var n="URLSearchParams"in e,r="Symbol"in e&&"iterator"in Symbol,i="FileReader"in e&&"Blob"in e&&function(){try{return new Blob,!0}catch(e){return!1}}(),s="FormData"in e,o="ArrayBuffer"in e;if(o)var a=["[object Int8Array]","[object Uint8Array]","[object Uint8ClampedArray]","[object Int16Array]","[object Uint16Array]","[object Int32Array]","[object Uint32Array]","[object Float32Array]","[object Float64Array]"],u=ArrayBuffer.isView||function(e){return e&&a.indexOf(Object.prototype.toString.call(e))>-1};function l(e){if("string"!=typeof e&&(e=String(e)),/[^a-z0-9\-#$%&'*+.^_`|~]/i.test(e))throw new TypeError("Invalid character in header field name");return e.toLowerCase()}function c(e){return"string"!=typeof e&&(e=String(e)),e}function h(e){var t={next:function(){var t=e.shift();return{done:void 0===t,value:t}}};return r&&(t[Symbol.iterator]=function(){return t}),t}function d(e){this.map={},e instanceof d?e.forEach((function(e,t){this.append(t,e)}),this):Array.isArray(e)?e.forEach((function(e){this.append(e[0],e[1])}),this):e&&Object.getOwnPropertyNames(e).forEach((function(t){this.append(t,e[t])}),this)}function p(e){if(e.bodyUsed)return Promise.reject(new TypeError("Already read"));e.bodyUsed=!0}function f(e){return new Promise((function(t,n){e.onload=function(){t(e.result)},e.onerror=function(){n(e.error)}}))}function y(e){var t=new FileReader,n=f(t);return t.readAsArrayBuffer(e),n}function b(e){if(e.slice)return e.slice(0);var t=new Uint8Array(e.byteLength);return t.set(new Uint8Array(e)),t.buffer}function v(){return this.bodyUsed=!1,this._initBody=function(e){var t;this._bodyInit=e,e?"string"==typeof e?this._bodyText=e:i&&Blob.prototype.isPrototypeOf(e)?this._bodyBlob=e:s&&FormData.prototype.isPrototypeOf(e)?this._bodyFormData=e:n&&URLSearchParams.prototype.isPrototypeOf(e)?this._bodyText=e.toString():o&&i&&((t=e)&&DataView.prototype.isPrototypeOf(t))?(this._bodyArrayBuffer=b(e.buffer),this._bodyInit=new Blob([this._bodyArrayBuffer])):o&&(ArrayBuffer.prototype.isPrototypeOf(e)||u(e))?this._bodyArrayBuffer=b(e):this._bodyText=e=Object.prototype.toString.call(e):this._bodyText="",this.headers.get("content-type")||("string"==typeof e?this.headers.set("content-type","text/plain;charset=UTF-8"):this._bodyBlob&&this._bodyBlob.type?this.headers.set("content-type",this._bodyBlob.type):n&&URLSearchParams.prototype.isPrototypeOf(e)&&this.headers.set("content-type","application/x-www-form-urlencoded;charset=UTF-8"))},i&&(this.blob=function(){var e=p(this);if(e)return e;if(this._bodyBlob)return Promise.resolve(this._bodyBlob);if(this._bodyArrayBuffer)return Promise.resolve(new Blob([this._bodyArrayBuffer]));if(this._bodyFormData)throw new Error("could not read FormData body as blob");return Promise.resolve(new Blob([this._bodyText]))},this.arrayBuffer=function(){return this._bodyArrayBuffer?p(this)||Promise.resolve(this._bodyArrayBuffer):this.blob().then(y)}),this.text=function(){var e,t,n,r=p(this);if(r)return r;if(this._bodyBlob)return e=this._bodyBlob,t=new FileReader,n=f(t),t.readAsText(e),n;if(this._bodyArrayBuffer)return Promise.resolve(function(e){for(var t=new Uint8Array(e),n=new Array(t.length),r=0;r<t.length;r++)n[r]=String.fromCharCode(t[r]);return n.join("")}(this._bodyArrayBuffer));if(this._bodyFormData)throw new Error("could not read FormData body as text");return Promise.resolve(this._bodyText)},s&&(this.formData=function(){return this.text().then(L)}),this.json=function(){return this.text().then(JSON.parse)},this}d.prototype.append=function(e,t){e=l(e),t=c(t);var n=this.map[e];this.map[e]=n?n+", "+t:t},d.prototype.delete=function(e){delete this.map[l(e)]},d.prototype.get=function(e){return e=l(e),this.has(e)?this.map[e]:null},d.prototype.has=function(e){return this.map.hasOwnProperty(l(e))},d.prototype.set=function(e,t){this.map[l(e)]=c(t)},d.prototype.forEach=function(e,t){for(var n in this.map)this.map.hasOwnProperty(n)&&e.call(t,this.map[n],n,this)},d.prototype.keys=function(){var e=[];return this.forEach((function(t,n){e.push(n)})),h(e)},d.prototype.values=function(){var e=[];return this.forEach((function(t){e.push(t)})),h(e)},d.prototype.entries=function(){var e=[];return this.forEach((function(t,n){e.push([n,t])})),h(e)},r&&(d.prototype[Symbol.iterator]=d.prototype.entries);var m=["DELETE","GET","HEAD","OPTIONS","POST","PUT"];function E(e,t){var n,r,i=(t=t||{}).body;if(e instanceof E){if(e.bodyUsed)throw new TypeError("Already read");this.url=e.url,this.credentials=e.credentials,t.headers||(this.headers=new d(e.headers)),this.method=e.method,this.mode=e.mode,this.signal=e.signal,i||null==e._bodyInit||(i=e._bodyInit,e.bodyUsed=!0)}else this.url=String(e);if(this.credentials=t.credentials||this.credentials||"same-origin",!t.headers&&this.headers||(this.headers=new d(t.headers)),this.method=(n=t.method||this.method||"GET",r=n.toUpperCase(),m.indexOf(r)>-1?r:n),this.mode=t.mode||this.mode||null,this.signal=t.signal||this.signal,this.referrer=null,("GET"===this.method||"HEAD"===this.method)&&i)throw new TypeError("Body not allowed for GET or HEAD requests");this._initBody(i)}function L(e){var t=new FormData;return e.trim().split("&").forEach((function(e){if(e){var n=e.split("="),r=n.shift().replace(/\+/g," "),i=n.join("=").replace(/\+/g," ");t.append(decodeURIComponent(r),decodeURIComponent(i))}})),t}function S(e,t){t||(t={}),this.type="default",this.status=void 0===t.status?200:t.status,this.ok=this.status>=200&&this.status<300,this.statusText="statusText"in t?t.statusText:"OK",this.headers=new d(t.headers),this.url=t.url||"",this._initBody(e)}E.prototype.clone=function(){return new E(this,{body:this._bodyInit})},v.call(E.prototype),v.call(S.prototype),S.prototype.clone=function(){return new S(this._bodyInit,{status:this.status,statusText:this.statusText,headers:new d(this.headers),url:this.url})},S.error=function(){var e=new S(null,{status:0,statusText:""});return e.type="error",e};var T=[301,302,303,307,308];S.redirect=function(e,t){if(-1===T.indexOf(t))throw new RangeError("Invalid status code");return new S(null,{status:t,headers:{location:e}})},t.DOMException=e.DOMException;try{new t.DOMException}catch(e){t.DOMException=function(e,t){this.message=e,this.name=t;var n=Error(e);this.stack=n.stack},t.DOMException.prototype=Object.create(Error.prototype),t.DOMException.prototype.constructor=t.DOMException}function _(e,n){return new Promise((function(r,s){var o=new E(e,n);if(o.signal&&o.signal.aborted)return s(new t.DOMException("Aborted","AbortError"));var a=new XMLHttpRequest;function u(){a.abort()}a.onload=function(){var e,t,n={status:a.status,statusText:a.statusText,headers:(e=a.getAllResponseHeaders()||"",t=new d,e.replace(/\r?\n[\t ]+/g," ").split(/\r?\n/).forEach((function(e){var n=e.split(":"),r=n.shift().trim();if(r){var i=n.join(":").trim();t.append(r,i)}})),t)};n.url="responseURL"in a?a.responseURL:n.headers.get("X-Request-URL");var i="response"in a?a.response:a.responseText;r(new S(i,n))},a.onerror=function(){s(new TypeError("Network request failed"))},a.ontimeout=function(){s(new TypeError("Network request failed"))},a.onabort=function(){s(new t.DOMException("Aborted","AbortError"))},a.open(o.method,o.url,!0),"include"===o.credentials?a.withCredentials=!0:"omit"===o.credentials&&(a.withCredentials=!1),"responseType"in a&&i&&(a.responseType="blob"),o.headers.forEach((function(e,t){a.setRequestHeader(t,e)})),o.signal&&(o.signal.addEventListener("abort",u),a.onreadystatechange=function(){4===a.readyState&&o.signal.removeEventListener("abort",u)}),a.send(void 0===o._bodyInit?null:o._bodyInit)}))}_.polyfill=!0,e.fetch||(e.fetch=_,e.Headers=d,e.Request=E,e.Response=S),t.Headers=d,t.Request=E,t.Response=S,t.fetch=_,Object.defineProperty(t,"__esModule",{value:!0})}({})}(r),r.fetch.ponyfill=!0,delete r.fetch.polyfill;var i=r;(t=i.fetch).default=i.fetch,t.fetch=i.fetch,t.Headers=i.Headers,t.Request=i.Request,t.Response=i.Response,e.exports=t},function(e,t,n){"use strict";var r,i=n(3),s=(r=i)&&r.__esModule?r:{default:r};document.addEventListener("DOMContentLoaded",(function(){new s.default}))},function(e,t,n){"use strict";Object.defineProperty(t,"__esModule",{value:!0});var r=function(){function e(e,t){for(var n=0;n<t.length;n++){var r=t[n];r.enumerable=r.enumerable||!1,r.configurable=!0,"value"in r&&(r.writable=!0),Object.defineProperty(e,r.key,r)}}return function(t,n,r){return n&&e(t.prototype,n),r&&e(t,r),t}}();function i(e){if(Array.isArray(e)){for(var t=0,n=Array(e.length);t<e.length;t++)n[t]=e[t];return n}return Array.from(e)}var s=function(){function e(){!function(e,t){if(!(e instanceof t))throw new TypeError("Cannot call a class as a function")}(this,e),this.OPTIONS_FORM_SELECTOR="#ctl-options",this.HEADER_SELECTOR=this.OPTIONS_FORM_SELECTOR+" h2",this.TABLE_SELECTOR=this.OPTIONS_FORM_SELECTOR+" table",this.SUBMIT_SELECTOR=this.OPTIONS_FORM_SELECTOR+" #submit",this.CURRENT_STUB_ID="ctl-current",this.CURRENT_NAV_TAB_CLASS="nav-tab-current",this.ACTIVE_NAV_TAB_CLASS="nav-tab-active",this.ACTIVE_TABLE_CLASS="active",this.EDIT_LABEL_ID="ctl-edit-label",this.EDIT_LABEL_ERROR_CLASS="ctl-edit-label-error",this.plusButton='<button type="button" aria-haspopup="true" aria-expanded="false" class="components-button block-editor-inserter__toggle has-icon" aria-label="Добавить блок"><svg width="24" height="24" xmlns="http://www.w3.org/2000/svg" viewBox="-2 -2 24 24" role="img" aria-hidden="true" focusable="false"><path d="M10 1c-5 0-9 4-9 9s4 9 9 9 9-4 9-9-4-9-9-9zm0 16c-3.9 0-7-3.1-7-7s3.1-7 7-7 7 3.1 7 7-3.1 7-7 7zm1-11H9v3H6v2h3v3h2v-3h3V9h-3V6zM10 1c-5 0-9 4-9 9s4 9 9 9 9-4 9-9-4-9-9-9zm0 16c-3.9 0-7-3.1-7-7s3.1-7 7-7 7 3.1 7 7-3.1 7-7 7zm1-11H9v3H6v2h3v3h2v-3h3V9h-3V6z"></path></svg></button>',this.PLUS_CLASS="ctl-plus",this.optionsForm=document.querySelector(this.OPTIONS_FORM_SELECTOR),this.tablesData=this.getTablesData(),this.submitButton=document.querySelector(this.SUBMIT_SELECTOR),this.optionsSaveSuccessMessage=Cyr2LatTablesObject.optionsSaveSuccessMessage,this.optionsSaveErrorMessage=Cyr2LatTablesObject.optionsSaveErrorMessage,this.addWrapper(),this.addMessageLines(),this.moveTabs(),this.addEditLabelInput(),this.hideTables(),this.bindEvents(),this.setSubmitStatus()}return r(e,[{key:"getHeaders",value:function(){return[].concat(i(document.querySelectorAll(this.HEADER_SELECTOR)))}},{key:"getActiveHeader",value:function(){return document.querySelector(this.HEADER_SELECTOR+"."+this.ACTIVE_NAV_TAB_CLASS)}},{key:"getActiveIndex",value:function(){return this.getActiveHeader().dataset.index}},{key:"getTables",value:function(){return[].concat(i(document.querySelectorAll(this.TABLE_SELECTOR)))}},{key:"getActiveTable",value:function(){return document.querySelector(this.TABLE_SELECTOR+"."+this.ACTIVE_TABLE_CLASS)}},{key:"getInputs",value:function(){return[].concat(i(document.querySelectorAll(this.OPTIONS_FORM_SELECTOR+" input")))}},{key:"getLabels",value:function(){return[].concat(i(document.querySelectorAll(this.OPTIONS_FORM_SELECTOR+" label")))}},{key:"getPlusButtons",value:function(){return[].concat(i(document.querySelectorAll(this.OPTIONS_FORM_SELECTOR+" ."+this.PLUS_CLASS)))}},{key:"isActiveTableChanged",value:function(){var e=this.getActiveIndex();return JSON.stringify(this.getActiveTableData())!==JSON.stringify(this.tablesData[e])}},{key:"setSubmitStatus",value:function(){this.submitButton.disabled=!this.isActiveTableChanged()}},{key:"saveActiveTable",value:function(){var e=this;if(this.isActiveTableChanged()){var t=this.getActiveTable(),n=document.createElement("form");return n.action=this.optionsForm.getAttribute("action"),n.method=this.optionsForm.method,n.appendChild(t.cloneNode(!0)),[].concat(i(t.querySelectorAll("input"))).map((function(e){return n.querySelector("#"+e.id).value=e.value})),[].concat(i(this.optionsForm.querySelectorAll('input[type="hidden"]'))).map((function(e){return n.appendChild(e.cloneNode(!0))})),document.body.appendChild(n),fetch(this.optionsForm.getAttribute("action"),{method:n.method,body:new URLSearchParams([].concat(i(new FormData(n))))}).then((function(t){return t.ok?(e.showMessage(e.successMessage,e.optionsSaveSuccessMessage),e.tablesData=e.getTablesData()):e.showMessage(e.errorMessage,e.optionsSaveErrorMessage),t.json()})).finally((function(){n.parentNode.removeChild(n),e.setSubmitStatus()}))}}},{key:"getTableData",value:function(e){var t=this,n=[].concat(i(e.querySelectorAll("input"))),r={};return n.forEach((function(e){var n=document.querySelector(t.OPTIONS_FORM_SELECTOR+' label[for="'+e.id+'"]');r[n.innerHTML]=e.value})),r}},{key:"getTablesData",value:function(){var e=this;return this.getTables().map((function(t){return e.getTableData(t)}))}},{key:"getActiveTableData",value:function(){return this.getTableData(this.getActiveTable())}},{key:"addWrapper",value:function(){this.wrapper=document.createElement("ul"),this.wrapper.classList.add("nav-tab-wrapper"),this.optionsForm.insertBefore(this.wrapper,this.optionsForm.firstChild)}},{key:"addMessageLine",value:function(e){var t=document.createElement("div");return t.id=e,this.optionsForm.insertBefore(t,this.optionsForm.firstChild),t}},{key:"addMessageLines",value:function(){this.successMessage=this.addMessageLine("ctl-success"),this.errorMessage=this.addMessageLine("ctl-error")}},{key:"moveTabs",value:function(){var e=document.querySelector(".ctl-settings-tabs");e&&this.optionsForm.insertBefore(e,this.optionsForm.firstChild)}},{key:"addEditLabelInput",value:function(){this.editLabelInput=document.createElement("input"),this.editLabelInput.id=this.EDIT_LABEL_ID,this.editLabelInput.style.display="none",document.body.appendChild(this.editLabelInput)}},{key:"hideEditLabelInput",value:function(){this.editLabelInput.style.display="none",this.editLabelInput.classList.remove(this.EDIT_LABEL_ERROR_CLASS),document.body.appendChild(this.editLabelInput)}},{key:"getLastCell",value:function(){return document.querySelector(this.OPTIONS_FORM_SELECTOR+" ."+this.ACTIVE_TABLE_CLASS+" ."+this.PLUS_CLASS).previousElementSibling}},{key:"addCell",value:function(){var e=this.getLastCell();e.parentElement.insertBefore(e.cloneNode(!0),e.nextElementSibling);var t=(e=this.getLastCell()).querySelector("label"),n=e.querySelector("input"),r=n.id.split("-"),i=r[0]+"-"+(parseInt(r[1])+1);t.htmlFor=i,t.innerHTML="",n.id=i,n.value="",n.setAttribute("value",""),this.replaceName(n,""),this.bindEvents(),this.editLabel(t)}},{key:"hideTables",value:function(){var e=this,t=0;this.getTables().map((function(n,r){n.classList.add("ctl-table"),e.CURRENT_STUB_ID===n.previousElementSibling.id&&(t=r,n.classList.add(e.ACTIVE_TABLE_CLASS));var i=document.createElement("div");return i.classList.add(e.PLUS_CLASS),i.innerHTML=e.plusButton,n.querySelector("td").appendChild(i),null})),this.getHeaders().map((function(n,r){return n.classList.add("nav-tab"),n.dataset.index=r,e.wrapper.appendChild(n),r===t&&(n.classList.add(e.CURRENT_NAV_TAB_CLASS),n.classList.add(e.ACTIVE_NAV_TAB_CLASS)),null}))}},{key:"bindEvents",value:function(){var e=this;this.getHeaders().map((function(t,n,r){return t.onclick=function(t){t.preventDefault();var n=t.target.dataset.index;if(n===e.getActiveIndex())return!1;e.saveActiveTable(),r.map((function(t){return t.classList.remove(e.ACTIVE_NAV_TAB_CLASS)})),r[n].classList.add(e.ACTIVE_NAV_TAB_CLASS);var i=e.getTables();return i.map((function(t){return t.classList.remove(e.ACTIVE_TABLE_CLASS)})),i[n].classList.add(e.ACTIVE_TABLE_CLASS),e.setSubmitStatus(),!1},null})),this.getInputs().map((function(t){return t.oninput=function(){e.setSubmitStatus()},null})),this.getLabels().map((function(t){return t.onclick=function(t){return t.preventDefault(),e.editLabel(t.target),!1},null})),this.editLabelInput.onblur=function(){e.saveLabel()},this.editLabelInput.onkeyup=function(t){"Escape"===t.key&&e.saveLabel(!0),"Enter"===t.key&&e.saveLabel()},this.getPlusButtons().map((function(t){return t.onclick=function(t){return t.preventDefault(),e.addCell(t.target),!1},null})),this.submitButton.onclick=function(t){return t.preventDefault(),e.saveActiveTable(),!1}}},{key:"editLabel",value:function(e){e.parentNode.appendChild(this.editLabelInput),this.editLabelInput.value=e.innerHTML,this.editLabelInput.classList.remove(this.EDIT_LABEL_ERROR_CLASS),this.editLabelInput.style.display="block",this.editLabelInput.focus()}},{key:"isUniqueLabel",value:function(e){return[].concat(i(this.getActiveTable().querySelectorAll("label"))).reduce((function(t,n){return t&&n.innerHTML!==e}),!0)}},{key:"saveLabel",value:function(){var e=arguments.length>0&&void 0!==arguments[0]&&arguments[0];if("none"!==this.editLabelInput.style.display){var t=this.editLabelInput.value.trim();if(""===t){var n=document.getElementById(this.EDIT_LABEL_ID).parentElement;return this.hideEditLabelInput(),n.parentNode.removeChild(n),void this.setSubmitStatus()}var r=this.editLabelInput.parentNode.querySelector("label");if(e||t===r.innerHTML)this.hideEditLabelInput();else if(this.isUniqueLabel(t)){var i=this.editLabelInput.parentNode.querySelector("input");this.hideEditLabelInput(),r.innerHTML=t,this.replaceName(i,t),this.setSubmitStatus()}else this.editLabelInput.classList.add(this.EDIT_LABEL_ERROR_CLASS)}}},{key:"replaceName",value:function(e,t){e.name=e.name.replace(/(.+\[.+])\[.*]/g,"$1["+t+"]")}},{key:"clearMessage",value:function(e){e.innerHTML="",e.classList.remove("active")}},{key:"clearMessages",value:function(){this.clearMessage(this.successMessage),this.clearMessage(this.errorMessage),clearTimeout(this.msgTimer)}},{key:"showMessage",value:function(e,t){var n=this;e.innerHTML=t,e.classList.add("active"),this.msgTimer=setTimeout((function(){n.clearMessages()}),5e3)}}]),e}();t.default=s}]);
2
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["webpack:///webpack/bootstrap","webpack:///./node_modules/cross-fetch/dist/browser-ponyfill.js","webpack:///./src/js/tables/app.js","webpack:///./src/js/tables/tables.js"],"names":["installedModules","__webpack_require__","moduleId","exports","module","i","l","modules","call","m","c","d","name","getter","o","Object","defineProperty","enumerable","get","r","Symbol","toStringTag","value","t","mode","__esModule","ns","create","key","bind","n","object","property","prototype","hasOwnProperty","p","s","global","self","this","__self__","F","fetch","DOMException","support","Blob","e","viewClasses","isArrayBufferView","ArrayBuffer","isView","obj","indexOf","toString","normalizeName","String","test","TypeError","toLowerCase","normalizeValue","iteratorFor","items","iterator","next","shift","done","undefined","Headers","headers","map","forEach","append","Array","isArray","header","getOwnPropertyNames","consumed","body","bodyUsed","Promise","reject","fileReaderReady","reader","resolve","onload","result","onerror","error","readBlobAsArrayBuffer","blob","FileReader","promise","readAsArrayBuffer","bufferClone","buf","slice","view","Uint8Array","byteLength","set","buffer","Body","_initBody","_bodyInit","_bodyText","isPrototypeOf","_bodyBlob","FormData","_bodyFormData","URLSearchParams","DataView","_bodyArrayBuffer","type","rejected","Error","arrayBuffer","then","text","readAsText","chars","length","fromCharCode","join","readArrayBufferAsText","formData","decode","json","JSON","parse","oldValue","has","callback","thisArg","keys","push","values","entries","methods","Request","input","options","method","upcased","url","credentials","signal","toUpperCase","referrer","form","trim","split","bytes","replace","decodeURIComponent","Response","bodyInit","status","ok","statusText","clone","response","redirectStatuses","redirect","RangeError","location","err","message","stack","constructor","init","request","aborted","xhr","XMLHttpRequest","abortXhr","abort","rawHeaders","getAllResponseHeaders","line","parts","responseURL","responseText","ontimeout","onabort","open","withCredentials","responseType","setRequestHeader","addEventListener","onreadystatechange","readyState","removeEventListener","send","polyfill","ponyfill","ctx","default","document","Tables","OPTIONS_FORM_SELECTOR","HEADER_SELECTOR","TABLE_SELECTOR","SUBMIT_SELECTOR","CURRENT_STUB_ID","CURRENT_NAV_TAB_CLASS","ACTIVE_NAV_TAB_CLASS","ACTIVE_TABLE_CLASS","EDIT_LABEL_ID","EDIT_LABEL_ERROR_CLASS","plusButton","PLUS_CLASS","optionsForm","querySelector","tablesData","getTablesData","submitButton","optionsSaveSuccessMessage","Cyr2LatTablesObject","optionsSaveErrorMessage","addWrapper","addMessageLines","moveTabs","addEditLabelInput","hideTables","bindEvents","setSubmitStatus","querySelectorAll","getActiveHeader","dataset","index","activeIndex","getActiveIndex","stringify","getActiveTableData","disabled","isActiveTableChanged","activeTable","getActiveTable","activeForm","createElement","action","getAttribute","appendChild","cloneNode","activeInputs","id","hiddenInputs","showMessage","successMessage","errorMessage","finally","parentNode","removeChild","table","inputs","data","label","innerHTML","getTables","getTableData","wrapper","classList","add","insertBefore","firstChild","addMessageLine","tabs","editLabelInput","style","display","remove","previousElementSibling","lastCell","getLastCell","parentElement","nextElementSibling","idArr","newId","parseInt","htmlFor","setAttribute","replaceName","editLabel","currentIndex","plus","getHeaders","onclick","event","preventDefault","target","saveActiveTable","aHeader","tables","getInputs","oninput","getLabels","onblur","saveLabel","onkeyup","getPlusButtons","addCell","focus","newValue","reduce","acc","cancel","editedCell","getElementById","hideEditLabelInput","isUniqueLabel","clearMessage","clearTimeout","msgTimer","el","setTimeout","clearMessages"],"mappings":"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","file":"tables/app.js","sourcesContent":[" \t// The module cache\n \tvar installedModules = {};\n\n \t// The require function\n \tfunction __webpack_require__(moduleId) {\n\n \t\t// Check if module is in cache\n \t\tif(installedModules[moduleId]) {\n \t\t\treturn installedModules[moduleId].exports;\n \t\t}\n \t\t// Create a new module (and put it into the cache)\n \t\tvar module = installedModules[moduleId] = {\n \t\t\ti: moduleId,\n \t\t\tl: false,\n \t\t\texports: {}\n \t\t};\n\n \t\t// Execute the module function\n \t\tmodules[moduleId].call(module.exports, module, module.exports, __webpack_require__);\n\n \t\t// Flag the module as loaded\n \t\tmodule.l = true;\n\n \t\t// Return the exports of the module\n \t\treturn module.exports;\n \t}\n\n\n \t// expose the modules object (__webpack_modules__)\n \t__webpack_require__.m = modules;\n\n \t// expose the module cache\n \t__webpack_require__.c = installedModules;\n\n \t// define getter function for harmony exports\n \t__webpack_require__.d = function(exports, name, getter) {\n \t\tif(!__webpack_require__.o(exports, name)) {\n \t\t\tObject.defineProperty(exports, name, { enumerable: true, get: getter });\n \t\t}\n \t};\n\n \t// define __esModule on exports\n \t__webpack_require__.r = function(exports) {\n \t\tif(typeof Symbol !== 'undefined' && Symbol.toStringTag) {\n \t\t\tObject.defineProperty(exports, Symbol.toStringTag, { value: 'Module' });\n \t\t}\n \t\tObject.defineProperty(exports, '__esModule', { value: true });\n \t};\n\n \t// create a fake namespace object\n \t// mode & 1: value is a module id, require it\n \t// mode & 2: merge all properties of value into the ns\n \t// mode & 4: return value when already ns object\n \t// mode & 8|1: behave like require\n \t__webpack_require__.t = function(value, mode) {\n \t\tif(mode & 1) value = __webpack_require__(value);\n \t\tif(mode & 8) return value;\n \t\tif((mode & 4) && typeof value === 'object' && value && value.__esModule) return value;\n \t\tvar ns = Object.create(null);\n \t\t__webpack_require__.r(ns);\n \t\tObject.defineProperty(ns, 'default', { enumerable: true, value: value });\n \t\tif(mode & 2 && typeof value != 'string') for(var key in value) __webpack_require__.d(ns, key, function(key) { return value[key]; }.bind(null, key));\n \t\treturn ns;\n \t};\n\n \t// getDefaultExport function for compatibility with non-harmony modules\n \t__webpack_require__.n = function(module) {\n \t\tvar getter = module && module.__esModule ?\n \t\t\tfunction getDefault() { return module['default']; } :\n \t\t\tfunction getModuleExports() { return module; };\n \t\t__webpack_require__.d(getter, 'a', getter);\n \t\treturn getter;\n \t};\n\n \t// Object.prototype.hasOwnProperty.call\n \t__webpack_require__.o = function(object, property) { return Object.prototype.hasOwnProperty.call(object, property); };\n\n \t// __webpack_public_path__\n \t__webpack_require__.p = \"\";\n\n\n \t// Load entry module and return exports\n \treturn __webpack_require__(__webpack_require__.s = 0);\n","var global = typeof self !== 'undefined' ? self : this;\nvar __self__ = (function () {\nfunction F() {\nthis.fetch = false;\nthis.DOMException = global.DOMException\n}\nF.prototype = global;\nreturn new F();\n})();\n(function(self) {\n\nvar irrelevant = (function (exports) {\n\n  var support = {\n    searchParams: 'URLSearchParams' in self,\n    iterable: 'Symbol' in self && 'iterator' in Symbol,\n    blob:\n      'FileReader' in self &&\n      'Blob' in self &&\n      (function() {\n        try {\n          new Blob();\n          return true\n        } catch (e) {\n          return false\n        }\n      })(),\n    formData: 'FormData' in self,\n    arrayBuffer: 'ArrayBuffer' in self\n  };\n\n  function isDataView(obj) {\n    return obj && DataView.prototype.isPrototypeOf(obj)\n  }\n\n  if (support.arrayBuffer) {\n    var viewClasses = [\n      '[object Int8Array]',\n      '[object Uint8Array]',\n      '[object Uint8ClampedArray]',\n      '[object Int16Array]',\n      '[object Uint16Array]',\n      '[object Int32Array]',\n      '[object Uint32Array]',\n      '[object Float32Array]',\n      '[object Float64Array]'\n    ];\n\n    var isArrayBufferView =\n      ArrayBuffer.isView ||\n      function(obj) {\n        return obj && viewClasses.indexOf(Object.prototype.toString.call(obj)) > -1\n      };\n  }\n\n  function normalizeName(name) {\n    if (typeof name !== 'string') {\n      name = String(name);\n    }\n    if (/[^a-z0-9\\-#$%&'*+.^_`|~]/i.test(name)) {\n      throw new TypeError('Invalid character in header field name')\n    }\n    return name.toLowerCase()\n  }\n\n  function normalizeValue(value) {\n    if (typeof value !== 'string') {\n      value = String(value);\n    }\n    return value\n  }\n\n  // Build a destructive iterator for the value list\n  function iteratorFor(items) {\n    var iterator = {\n      next: function() {\n        var value = items.shift();\n        return {done: value === undefined, value: value}\n      }\n    };\n\n    if (support.iterable) {\n      iterator[Symbol.iterator] = function() {\n        return iterator\n      };\n    }\n\n    return iterator\n  }\n\n  function Headers(headers) {\n    this.map = {};\n\n    if (headers instanceof Headers) {\n      headers.forEach(function(value, name) {\n        this.append(name, value);\n      }, this);\n    } else if (Array.isArray(headers)) {\n      headers.forEach(function(header) {\n        this.append(header[0], header[1]);\n      }, this);\n    } else if (headers) {\n      Object.getOwnPropertyNames(headers).forEach(function(name) {\n        this.append(name, headers[name]);\n      }, this);\n    }\n  }\n\n  Headers.prototype.append = function(name, value) {\n    name = normalizeName(name);\n    value = normalizeValue(value);\n    var oldValue = this.map[name];\n    this.map[name] = oldValue ? oldValue + ', ' + value : value;\n  };\n\n  Headers.prototype['delete'] = function(name) {\n    delete this.map[normalizeName(name)];\n  };\n\n  Headers.prototype.get = function(name) {\n    name = normalizeName(name);\n    return this.has(name) ? this.map[name] : null\n  };\n\n  Headers.prototype.has = function(name) {\n    return this.map.hasOwnProperty(normalizeName(name))\n  };\n\n  Headers.prototype.set = function(name, value) {\n    this.map[normalizeName(name)] = normalizeValue(value);\n  };\n\n  Headers.prototype.forEach = function(callback, thisArg) {\n    for (var name in this.map) {\n      if (this.map.hasOwnProperty(name)) {\n        callback.call(thisArg, this.map[name], name, this);\n      }\n    }\n  };\n\n  Headers.prototype.keys = function() {\n    var items = [];\n    this.forEach(function(value, name) {\n      items.push(name);\n    });\n    return iteratorFor(items)\n  };\n\n  Headers.prototype.values = function() {\n    var items = [];\n    this.forEach(function(value) {\n      items.push(value);\n    });\n    return iteratorFor(items)\n  };\n\n  Headers.prototype.entries = function() {\n    var items = [];\n    this.forEach(function(value, name) {\n      items.push([name, value]);\n    });\n    return iteratorFor(items)\n  };\n\n  if (support.iterable) {\n    Headers.prototype[Symbol.iterator] = Headers.prototype.entries;\n  }\n\n  function consumed(body) {\n    if (body.bodyUsed) {\n      return Promise.reject(new TypeError('Already read'))\n    }\n    body.bodyUsed = true;\n  }\n\n  function fileReaderReady(reader) {\n    return new Promise(function(resolve, reject) {\n      reader.onload = function() {\n        resolve(reader.result);\n      };\n      reader.onerror = function() {\n        reject(reader.error);\n      };\n    })\n  }\n\n  function readBlobAsArrayBuffer(blob) {\n    var reader = new FileReader();\n    var promise = fileReaderReady(reader);\n    reader.readAsArrayBuffer(blob);\n    return promise\n  }\n\n  function readBlobAsText(blob) {\n    var reader = new FileReader();\n    var promise = fileReaderReady(reader);\n    reader.readAsText(blob);\n    return promise\n  }\n\n  function readArrayBufferAsText(buf) {\n    var view = new Uint8Array(buf);\n    var chars = new Array(view.length);\n\n    for (var i = 0; i < view.length; i++) {\n      chars[i] = String.fromCharCode(view[i]);\n    }\n    return chars.join('')\n  }\n\n  function bufferClone(buf) {\n    if (buf.slice) {\n      return buf.slice(0)\n    } else {\n      var view = new Uint8Array(buf.byteLength);\n      view.set(new Uint8Array(buf));\n      return view.buffer\n    }\n  }\n\n  function Body() {\n    this.bodyUsed = false;\n\n    this._initBody = function(body) {\n      this._bodyInit = body;\n      if (!body) {\n        this._bodyText = '';\n      } else if (typeof body === 'string') {\n        this._bodyText = body;\n      } else if (support.blob && Blob.prototype.isPrototypeOf(body)) {\n        this._bodyBlob = body;\n      } else if (support.formData && FormData.prototype.isPrototypeOf(body)) {\n        this._bodyFormData = body;\n      } else if (support.searchParams && URLSearchParams.prototype.isPrototypeOf(body)) {\n        this._bodyText = body.toString();\n      } else if (support.arrayBuffer && support.blob && isDataView(body)) {\n        this._bodyArrayBuffer = bufferClone(body.buffer);\n        // IE 10-11 can't handle a DataView body.\n        this._bodyInit = new Blob([this._bodyArrayBuffer]);\n      } else if (support.arrayBuffer && (ArrayBuffer.prototype.isPrototypeOf(body) || isArrayBufferView(body))) {\n        this._bodyArrayBuffer = bufferClone(body);\n      } else {\n        this._bodyText = body = Object.prototype.toString.call(body);\n      }\n\n      if (!this.headers.get('content-type')) {\n        if (typeof body === 'string') {\n          this.headers.set('content-type', 'text/plain;charset=UTF-8');\n        } else if (this._bodyBlob && this._bodyBlob.type) {\n          this.headers.set('content-type', this._bodyBlob.type);\n        } else if (support.searchParams && URLSearchParams.prototype.isPrototypeOf(body)) {\n          this.headers.set('content-type', 'application/x-www-form-urlencoded;charset=UTF-8');\n        }\n      }\n    };\n\n    if (support.blob) {\n      this.blob = function() {\n        var rejected = consumed(this);\n        if (rejected) {\n          return rejected\n        }\n\n        if (this._bodyBlob) {\n          return Promise.resolve(this._bodyBlob)\n        } else if (this._bodyArrayBuffer) {\n          return Promise.resolve(new Blob([this._bodyArrayBuffer]))\n        } else if (this._bodyFormData) {\n          throw new Error('could not read FormData body as blob')\n        } else {\n          return Promise.resolve(new Blob([this._bodyText]))\n        }\n      };\n\n      this.arrayBuffer = function() {\n        if (this._bodyArrayBuffer) {\n          return consumed(this) || Promise.resolve(this._bodyArrayBuffer)\n        } else {\n          return this.blob().then(readBlobAsArrayBuffer)\n        }\n      };\n    }\n\n    this.text = function() {\n      var rejected = consumed(this);\n      if (rejected) {\n        return rejected\n      }\n\n      if (this._bodyBlob) {\n        return readBlobAsText(this._bodyBlob)\n      } else if (this._bodyArrayBuffer) {\n        return Promise.resolve(readArrayBufferAsText(this._bodyArrayBuffer))\n      } else if (this._bodyFormData) {\n        throw new Error('could not read FormData body as text')\n      } else {\n        return Promise.resolve(this._bodyText)\n      }\n    };\n\n    if (support.formData) {\n      this.formData = function() {\n        return this.text().then(decode)\n      };\n    }\n\n    this.json = function() {\n      return this.text().then(JSON.parse)\n    };\n\n    return this\n  }\n\n  // HTTP methods whose capitalization should be normalized\n  var methods = ['DELETE', 'GET', 'HEAD', 'OPTIONS', 'POST', 'PUT'];\n\n  function normalizeMethod(method) {\n    var upcased = method.toUpperCase();\n    return methods.indexOf(upcased) > -1 ? upcased : method\n  }\n\n  function Request(input, options) {\n    options = options || {};\n    var body = options.body;\n\n    if (input instanceof Request) {\n      if (input.bodyUsed) {\n        throw new TypeError('Already read')\n      }\n      this.url = input.url;\n      this.credentials = input.credentials;\n      if (!options.headers) {\n        this.headers = new Headers(input.headers);\n      }\n      this.method = input.method;\n      this.mode = input.mode;\n      this.signal = input.signal;\n      if (!body && input._bodyInit != null) {\n        body = input._bodyInit;\n        input.bodyUsed = true;\n      }\n    } else {\n      this.url = String(input);\n    }\n\n    this.credentials = options.credentials || this.credentials || 'same-origin';\n    if (options.headers || !this.headers) {\n      this.headers = new Headers(options.headers);\n    }\n    this.method = normalizeMethod(options.method || this.method || 'GET');\n    this.mode = options.mode || this.mode || null;\n    this.signal = options.signal || this.signal;\n    this.referrer = null;\n\n    if ((this.method === 'GET' || this.method === 'HEAD') && body) {\n      throw new TypeError('Body not allowed for GET or HEAD requests')\n    }\n    this._initBody(body);\n  }\n\n  Request.prototype.clone = function() {\n    return new Request(this, {body: this._bodyInit})\n  };\n\n  function decode(body) {\n    var form = new FormData();\n    body\n      .trim()\n      .split('&')\n      .forEach(function(bytes) {\n        if (bytes) {\n          var split = bytes.split('=');\n          var name = split.shift().replace(/\\+/g, ' ');\n          var value = split.join('=').replace(/\\+/g, ' ');\n          form.append(decodeURIComponent(name), decodeURIComponent(value));\n        }\n      });\n    return form\n  }\n\n  function parseHeaders(rawHeaders) {\n    var headers = new Headers();\n    // Replace instances of \\r\\n and \\n followed by at least one space or horizontal tab with a space\n    // https://tools.ietf.org/html/rfc7230#section-3.2\n    var preProcessedHeaders = rawHeaders.replace(/\\r?\\n[\\t ]+/g, ' ');\n    preProcessedHeaders.split(/\\r?\\n/).forEach(function(line) {\n      var parts = line.split(':');\n      var key = parts.shift().trim();\n      if (key) {\n        var value = parts.join(':').trim();\n        headers.append(key, value);\n      }\n    });\n    return headers\n  }\n\n  Body.call(Request.prototype);\n\n  function Response(bodyInit, options) {\n    if (!options) {\n      options = {};\n    }\n\n    this.type = 'default';\n    this.status = options.status === undefined ? 200 : options.status;\n    this.ok = this.status >= 200 && this.status < 300;\n    this.statusText = 'statusText' in options ? options.statusText : 'OK';\n    this.headers = new Headers(options.headers);\n    this.url = options.url || '';\n    this._initBody(bodyInit);\n  }\n\n  Body.call(Response.prototype);\n\n  Response.prototype.clone = function() {\n    return new Response(this._bodyInit, {\n      status: this.status,\n      statusText: this.statusText,\n      headers: new Headers(this.headers),\n      url: this.url\n    })\n  };\n\n  Response.error = function() {\n    var response = new Response(null, {status: 0, statusText: ''});\n    response.type = 'error';\n    return response\n  };\n\n  var redirectStatuses = [301, 302, 303, 307, 308];\n\n  Response.redirect = function(url, status) {\n    if (redirectStatuses.indexOf(status) === -1) {\n      throw new RangeError('Invalid status code')\n    }\n\n    return new Response(null, {status: status, headers: {location: url}})\n  };\n\n  exports.DOMException = self.DOMException;\n  try {\n    new exports.DOMException();\n  } catch (err) {\n    exports.DOMException = function(message, name) {\n      this.message = message;\n      this.name = name;\n      var error = Error(message);\n      this.stack = error.stack;\n    };\n    exports.DOMException.prototype = Object.create(Error.prototype);\n    exports.DOMException.prototype.constructor = exports.DOMException;\n  }\n\n  function fetch(input, init) {\n    return new Promise(function(resolve, reject) {\n      var request = new Request(input, init);\n\n      if (request.signal && request.signal.aborted) {\n        return reject(new exports.DOMException('Aborted', 'AbortError'))\n      }\n\n      var xhr = new XMLHttpRequest();\n\n      function abortXhr() {\n        xhr.abort();\n      }\n\n      xhr.onload = function() {\n        var options = {\n          status: xhr.status,\n          statusText: xhr.statusText,\n          headers: parseHeaders(xhr.getAllResponseHeaders() || '')\n        };\n        options.url = 'responseURL' in xhr ? xhr.responseURL : options.headers.get('X-Request-URL');\n        var body = 'response' in xhr ? xhr.response : xhr.responseText;\n        resolve(new Response(body, options));\n      };\n\n      xhr.onerror = function() {\n        reject(new TypeError('Network request failed'));\n      };\n\n      xhr.ontimeout = function() {\n        reject(new TypeError('Network request failed'));\n      };\n\n      xhr.onabort = function() {\n        reject(new exports.DOMException('Aborted', 'AbortError'));\n      };\n\n      xhr.open(request.method, request.url, true);\n\n      if (request.credentials === 'include') {\n        xhr.withCredentials = true;\n      } else if (request.credentials === 'omit') {\n        xhr.withCredentials = false;\n      }\n\n      if ('responseType' in xhr && support.blob) {\n        xhr.responseType = 'blob';\n      }\n\n      request.headers.forEach(function(value, name) {\n        xhr.setRequestHeader(name, value);\n      });\n\n      if (request.signal) {\n        request.signal.addEventListener('abort', abortXhr);\n\n        xhr.onreadystatechange = function() {\n          // DONE (success or failure)\n          if (xhr.readyState === 4) {\n            request.signal.removeEventListener('abort', abortXhr);\n          }\n        };\n      }\n\n      xhr.send(typeof request._bodyInit === 'undefined' ? null : request._bodyInit);\n    })\n  }\n\n  fetch.polyfill = true;\n\n  if (!self.fetch) {\n    self.fetch = fetch;\n    self.Headers = Headers;\n    self.Request = Request;\n    self.Response = Response;\n  }\n\n  exports.Headers = Headers;\n  exports.Request = Request;\n  exports.Response = Response;\n  exports.fetch = fetch;\n\n  Object.defineProperty(exports, '__esModule', { value: true });\n\n  return exports;\n\n}({}));\n})(__self__);\n__self__.fetch.ponyfill = true;\n// Remove \"polyfill\" property added by whatwg-fetch\ndelete __self__.fetch.polyfill;\n// Choose between native implementation (global) or custom implementation (__self__)\n// var ctx = global.fetch ? global : __self__;\nvar ctx = __self__; // this line disable service worker support temporarily\nexports = ctx.fetch // To enable: import fetch from 'cross-fetch'\nexports.default = ctx.fetch // For TypeScript consumers without esModuleInterop.\nexports.fetch = ctx.fetch // To enable: import {fetch} from 'cross-fetch'\nexports.Headers = ctx.Headers\nexports.Request = ctx.Request\nexports.Response = ctx.Response\nmodule.exports = exports\n","/**\n * @file Tables Application.\n */\nimport Tables from './tables';\n\ndocument.addEventListener( 'DOMContentLoaded', () => {\n\tnew Tables();\n} );\n","/**\n * @file class Tables.\n */\n\n/* global Cyr2LatTablesObject */\n\nclass Tables {\n\t/**\n\t * Class constructor.\n\t */\n\tconstructor() {\n\t\tthis.OPTIONS_FORM_SELECTOR = '#ctl-options';\n\t\tthis.HEADER_SELECTOR = this.OPTIONS_FORM_SELECTOR + ' h2';\n\t\tthis.TABLE_SELECTOR = this.OPTIONS_FORM_SELECTOR + ' table';\n\t\tthis.SUBMIT_SELECTOR = this.OPTIONS_FORM_SELECTOR + ' #submit';\n\t\tthis.CURRENT_STUB_ID = 'ctl-current';\n\t\tthis.CURRENT_NAV_TAB_CLASS = 'nav-tab-current';\n\t\tthis.ACTIVE_NAV_TAB_CLASS = 'nav-tab-active';\n\t\tthis.ACTIVE_TABLE_CLASS = 'active';\n\t\tthis.EDIT_LABEL_ID = 'ctl-edit-label';\n\t\tthis.EDIT_LABEL_ERROR_CLASS = 'ctl-edit-label-error';\n\t\tthis.plusButton =\n\t\t\t'<button type=\"button\" aria-haspopup=\"true\" aria-expanded=\"false\" class=\"components-button block-editor-inserter__toggle has-icon\" aria-label=\"Добавить блок\">' +\n\t\t\t'<svg width=\"24\" height=\"24\" xmlns=\"http://www.w3.org/2000/svg\" viewBox=\"-2 -2 24 24\" role=\"img\" aria-hidden=\"true\" focusable=\"false\">' +\n\t\t\t'<path d=\"M10 1c-5 0-9 4-9 9s4 9 9 9 9-4 9-9-4-9-9-9zm0 16c-3.9 0-7-3.1-7-7s3.1-7 7-7 7 3.1 7 7-3.1 7-7 7zm1-11H9v3H6v2h3v3h2v-3h3V9h-3V6zM10 1c-5 0-9 4-9 9s4 9 9 9 9-4 9-9-4-9-9-9zm0 16c-3.9 0-7-3.1-7-7s3.1-7 7-7 7 3.1 7 7-3.1 7-7 7zm1-11H9v3H6v2h3v3h2v-3h3V9h-3V6z\">' +\n\t\t\t'</path>' +\n\t\t\t'</svg>' +\n\t\t\t'</button>';\n\t\tthis.PLUS_CLASS = 'ctl-plus';\n\n\t\tthis.optionsForm = document.querySelector( this.OPTIONS_FORM_SELECTOR );\n\t\tthis.tablesData = this.getTablesData();\n\t\tthis.submitButton = document.querySelector( this.SUBMIT_SELECTOR );\n\n\t\t// Copy to class properties, otherwise eslint marks some properties of global object as unresolved.\n\t\tthis.optionsSaveSuccessMessage =\n\t\t\tCyr2LatTablesObject.optionsSaveSuccessMessage;\n\t\tthis.optionsSaveErrorMessage =\n\t\t\tCyr2LatTablesObject.optionsSaveErrorMessage;\n\n\t\tthis.addWrapper();\n\t\tthis.addMessageLines();\n\t\tthis.moveTabs();\n\t\tthis.addEditLabelInput();\n\t\tthis.hideTables();\n\t\tthis.bindEvents();\n\t\tthis.setSubmitStatus();\n\t}\n\n\t/**\n\t * Get headers.\n\t *\n\t * @return {*[]} Headers.\n\t */\n\tgetHeaders() {\n\t\treturn [ ...document.querySelectorAll( this.HEADER_SELECTOR ) ];\n\t}\n\n\t/**\n\t * Get active header.\n\t *\n\t * @return {Element} Active header.\n\t */\n\tgetActiveHeader() {\n\t\treturn document.querySelector(\n\t\t\tthis.HEADER_SELECTOR + '.' + this.ACTIVE_NAV_TAB_CLASS\n\t\t);\n\t}\n\n\t/**\n\t * Get active index.\n\t *\n\t * @return {*} Active index.\n\t */\n\tgetActiveIndex() {\n\t\treturn this.getActiveHeader().dataset.index;\n\t}\n\n\t/**\n\t * Get tables.\n\t *\n\t * @return {*[]} Tables.\n\t */\n\tgetTables() {\n\t\treturn [ ...document.querySelectorAll( this.TABLE_SELECTOR ) ];\n\t}\n\n\t/**\n\t * Get active table.\n\t *\n\t * @return {Element} Active table.\n\t */\n\tgetActiveTable() {\n\t\treturn document.querySelector(\n\t\t\tthis.TABLE_SELECTOR + '.' + this.ACTIVE_TABLE_CLASS\n\t\t);\n\t}\n\n\t/**\n\t * Get inputs.\n\t *\n\t * @return {*[]} Inputs.\n\t */\n\tgetInputs() {\n\t\treturn [\n\t\t\t...document.querySelectorAll(\n\t\t\t\tthis.OPTIONS_FORM_SELECTOR + ' input'\n\t\t\t),\n\t\t];\n\t}\n\n\t/**\n\t * Get labels.\n\t *\n\t * @return {*[]} Labels.\n\t */\n\tgetLabels() {\n\t\treturn [\n\t\t\t...document.querySelectorAll(\n\t\t\t\tthis.OPTIONS_FORM_SELECTOR + ' label'\n\t\t\t),\n\t\t];\n\t}\n\n\t/**\n\t * Get plus buttons.\n\t *\n\t * @return {*[]} Plus buttons.\n\t */\n\tgetPlusButtons() {\n\t\treturn [\n\t\t\t...document.querySelectorAll(\n\t\t\t\tthis.OPTIONS_FORM_SELECTOR + ' .' + this.PLUS_CLASS\n\t\t\t),\n\t\t];\n\t}\n\n\t/**\n\t * Check of active table was changed.\n\t *\n\t * @return {boolean} If active table was changed.\n\t */\n\tisActiveTableChanged() {\n\t\tconst activeIndex = this.getActiveIndex();\n\n\t\treturn (\n\t\t\tJSON.stringify( this.getActiveTableData() ) !==\n\t\t\tJSON.stringify( this.tablesData[ activeIndex ] )\n\t\t);\n\t}\n\n\t/**\n\t * Set status of submit button.\n\t */\n\tsetSubmitStatus() {\n\t\tthis.submitButton.disabled = ! this.isActiveTableChanged();\n\t}\n\n\t/**\n\t * Save active table.\n\t */\n\tsaveActiveTable() {\n\t\tif ( ! this.isActiveTableChanged() ) {\n\t\t\treturn;\n\t\t}\n\n\t\tconst activeTable = this.getActiveTable();\n\n\t\tconst activeForm = document.createElement( 'form' );\n\t\tactiveForm.action = this.optionsForm.getAttribute( 'action' );\n\t\tactiveForm.method = this.optionsForm.method;\n\t\tactiveForm.appendChild( activeTable.cloneNode( true ) );\n\n\t\tconst activeInputs = [ ...activeTable.querySelectorAll( 'input' ) ];\n\t\tactiveInputs.map( ( input ) => {\n\t\t\treturn ( activeForm.querySelector( '#' + input.id ).value =\n\t\t\t\tinput.value );\n\t\t} );\n\n\t\tconst hiddenInputs = [\n\t\t\t...this.optionsForm.querySelectorAll( 'input[type=\"hidden\"]' ),\n\t\t];\n\t\thiddenInputs.map( ( input ) => {\n\t\t\treturn activeForm.appendChild( input.cloneNode( true ) );\n\t\t} );\n\t\tdocument.body.appendChild( activeForm );\n\n\t\treturn fetch( this.optionsForm.getAttribute( 'action' ), {\n\t\t\tmethod: activeForm.method,\n\t\t\tbody: new URLSearchParams( [ ...new FormData( activeForm ) ] ),\n\t\t} )\n\t\t\t.then( ( response ) => {\n\t\t\t\tif ( response.ok ) {\n\t\t\t\t\tthis.showMessage(\n\t\t\t\t\t\tthis.successMessage,\n\t\t\t\t\t\tthis.optionsSaveSuccessMessage\n\t\t\t\t\t);\n\t\t\t\t\tthis.tablesData = this.getTablesData();\n\t\t\t\t} else {\n\t\t\t\t\tthis.showMessage(\n\t\t\t\t\t\tthis.errorMessage,\n\t\t\t\t\t\tthis.optionsSaveErrorMessage\n\t\t\t\t\t);\n\t\t\t\t}\n\n\t\t\t\treturn response.json();\n\t\t\t} )\n\t\t\t.finally( () => {\n\t\t\t\tactiveForm.parentNode.removeChild( activeForm );\n\t\t\t\tthis.setSubmitStatus();\n\t\t\t} );\n\t}\n\n\t/**\n\t * Get table data.\n\t *\n\t * @param {Element} table Table.\n\t * @return {{}[]} Table data.\n\t */\n\tgetTableData( table ) {\n\t\tconst inputs = [ ...table.querySelectorAll( 'input' ) ];\n\n\t\tconst data = {};\n\t\tinputs.forEach( ( input ) => {\n\t\t\tconst label = document.querySelector(\n\t\t\t\tthis.OPTIONS_FORM_SELECTOR + ' label[for=\"' + input.id + '\"]'\n\t\t\t);\n\n\t\t\tdata[ label.innerHTML ] = input.value;\n\t\t} );\n\n\t\treturn data;\n\t}\n\n\t/**\n\t * Get data from all tables.\n\t *\n\t * @return {{}[][]} Data from all tables.\n\t */\n\tgetTablesData() {\n\t\treturn this.getTables().map( ( table ) => {\n\t\t\treturn this.getTableData( table );\n\t\t} );\n\t}\n\n\t/**\n\t * Get active table data.\n\t *\n\t * @return {{}[]} Active table data.\n\t */\n\tgetActiveTableData() {\n\t\treturn this.getTableData( this.getActiveTable() );\n\t}\n\n\t/**\n\t * Add wrapper.\n\t */\n\taddWrapper() {\n\t\tthis.wrapper = document.createElement( 'ul' );\n\t\tthis.wrapper.classList.add( 'nav-tab-wrapper' );\n\t\tthis.optionsForm.insertBefore(\n\t\t\tthis.wrapper,\n\t\t\tthis.optionsForm.firstChild\n\t\t);\n\t}\n\n\t/**\n\t * Add message line.\n\t *\n\t * @param {string} id Message id.\n\t * @return {HTMLDivElement} Message line.\n\t */\n\taddMessageLine( id ) {\n\t\tconst message = document.createElement( 'div' );\n\t\tmessage.id = id;\n\t\tthis.optionsForm.insertBefore( message, this.optionsForm.firstChild );\n\n\t\treturn message;\n\t}\n\n\t/**\n\t * Add success and error message lines.\n\t */\n\taddMessageLines() {\n\t\tthis.successMessage = this.addMessageLine( 'ctl-success' );\n\t\tthis.errorMessage = this.addMessageLine( 'ctl-error' );\n\t}\n\n\t/**\n\t * Make tabs the first element in the form.\n\t */\n\tmoveTabs() {\n\t\tconst tabs = document.querySelector( '.ctl-settings-tabs' );\n\t\tif ( tabs ) {\n\t\t\tthis.optionsForm.insertBefore( tabs, this.optionsForm.firstChild );\n\t\t}\n\t}\n\n\t/**\n\t * Add edit label input.\n\t */\n\taddEditLabelInput() {\n\t\tthis.editLabelInput = document.createElement( 'input' );\n\t\tthis.editLabelInput.id = this.EDIT_LABEL_ID;\n\t\tthis.editLabelInput.style.display = 'none';\n\t\tdocument.body.appendChild( this.editLabelInput );\n\t}\n\n\t/**\n\t * Hide edit label input.\n\t */\n\thideEditLabelInput() {\n\t\tthis.editLabelInput.style.display = 'none';\n\t\tthis.editLabelInput.classList.remove( this.EDIT_LABEL_ERROR_CLASS );\n\t\tdocument.body.appendChild( this.editLabelInput );\n\t}\n\n\t/**\n\t * Get last cell in active table.\n\t *\n\t * @return {Element} Last cell.\n\t */\n\tgetLastCell() {\n\t\treturn document.querySelector(\n\t\t\tthis.OPTIONS_FORM_SELECTOR +\n\t\t\t\t' .' +\n\t\t\t\tthis.ACTIVE_TABLE_CLASS +\n\t\t\t\t' .' +\n\t\t\t\tthis.PLUS_CLASS\n\t\t).previousElementSibling;\n\t}\n\n\t/**\n\t * Add new cell to the active table.\n\t */\n\taddCell() {\n\t\tlet lastCell = this.getLastCell();\n\t\tlastCell.parentElement.insertBefore(\n\t\t\tlastCell.cloneNode( true ),\n\t\t\tlastCell.nextElementSibling\n\t\t);\n\n\t\tlastCell = this.getLastCell();\n\t\tconst label = lastCell.querySelector( 'label' );\n\t\tconst input = lastCell.querySelector( 'input' );\n\n\t\tconst idArr = input.id.split( '-' );\n\t\tconst newId = idArr[ 0 ] + '-' + ( parseInt( idArr[ 1 ] ) + 1 );\n\n\t\tlabel.htmlFor = newId;\n\t\tlabel.innerHTML = '';\n\n\t\tinput.id = newId;\n\t\tinput.value = '';\n\t\tinput.setAttribute( 'value', '' );\n\t\tthis.replaceName( input, '' );\n\n\t\tthis.bindEvents();\n\n\t\tthis.editLabel( label );\n\t}\n\n\t/**\n\t * Hide conversion tables except the first one.\n\t * Create navigation tabs.\n\t */\n\thideTables() {\n\t\tlet currentIndex = 0;\n\n\t\tthis.getTables().map( ( table, index ) => {\n\t\t\ttable.classList.add( 'ctl-table' );\n\n\t\t\tif ( this.CURRENT_STUB_ID === table.previousElementSibling.id ) {\n\t\t\t\tcurrentIndex = index;\n\t\t\t\ttable.classList.add( this.ACTIVE_TABLE_CLASS );\n\t\t\t}\n\n\t\t\tconst plus = document.createElement( 'div' );\n\t\t\tplus.classList.add( this.PLUS_CLASS );\n\t\t\tplus.innerHTML = this.plusButton;\n\t\t\ttable.querySelector( 'td' ).appendChild( plus );\n\n\t\t\treturn null;\n\t\t} );\n\n\t\tthis.getHeaders().map( ( header, index ) => {\n\t\t\theader.classList.add( 'nav-tab' );\n\t\t\theader.dataset.index = index;\n\n\t\t\tthis.wrapper.appendChild( header );\n\t\t\tif ( index === currentIndex ) {\n\t\t\t\theader.classList.add( this.CURRENT_NAV_TAB_CLASS );\n\t\t\t\theader.classList.add( this.ACTIVE_NAV_TAB_CLASS );\n\t\t\t}\n\n\t\t\treturn null;\n\t\t} );\n\t}\n\n\t/**\n\t * Bind events to methods.\n\t */\n\tbindEvents() {\n\t\tthis.getHeaders().map( ( header, i, headers ) => {\n\t\t\theader.onclick = ( event ) => {\n\t\t\t\tevent.preventDefault();\n\n\t\t\t\tconst index = event.target.dataset.index;\n\t\t\t\tconst activeIndex = this.getActiveIndex();\n\n\t\t\t\tif ( index === activeIndex ) {\n\t\t\t\t\treturn false;\n\t\t\t\t}\n\n\t\t\t\tthis.saveActiveTable();\n\n\t\t\t\theaders.map( ( aHeader ) => {\n\t\t\t\t\treturn aHeader.classList.remove(\n\t\t\t\t\t\tthis.ACTIVE_NAV_TAB_CLASS\n\t\t\t\t\t);\n\t\t\t\t} );\n\t\t\t\theaders[ index ].classList.add( this.ACTIVE_NAV_TAB_CLASS );\n\n\t\t\t\tconst tables = this.getTables();\n\t\t\t\ttables.map( ( table ) => {\n\t\t\t\t\treturn table.classList.remove( this.ACTIVE_TABLE_CLASS );\n\t\t\t\t} );\n\t\t\t\ttables[ index ].classList.add( this.ACTIVE_TABLE_CLASS );\n\n\t\t\t\tthis.setSubmitStatus();\n\n\t\t\t\treturn false;\n\t\t\t};\n\n\t\t\treturn null;\n\t\t} );\n\n\t\tthis.getInputs().map( ( input ) => {\n\t\t\tinput.oninput = () => {\n\t\t\t\tthis.setSubmitStatus();\n\t\t\t};\n\n\t\t\treturn null;\n\t\t} );\n\n\t\tthis.getLabels().map( ( label ) => {\n\t\t\tlabel.onclick = ( event ) => {\n\t\t\t\tevent.preventDefault();\n\t\t\t\tthis.editLabel( event.target );\n\t\t\t\treturn false;\n\t\t\t};\n\n\t\t\treturn null;\n\t\t} );\n\n\t\tthis.editLabelInput.onblur = () => {\n\t\t\tthis.saveLabel();\n\t\t};\n\n\t\tthis.editLabelInput.onkeyup = ( e ) => {\n\t\t\tif ( 'Escape' === e.key ) {\n\t\t\t\tthis.saveLabel( true );\n\t\t\t}\n\n\t\t\tif ( 'Enter' === e.key ) {\n\t\t\t\tthis.saveLabel();\n\t\t\t}\n\t\t};\n\n\t\tthis.getPlusButtons().map( ( plus ) => {\n\t\t\tplus.onclick = ( event ) => {\n\t\t\t\tevent.preventDefault();\n\t\t\t\tthis.addCell( event.target );\n\t\t\t\treturn false;\n\t\t\t};\n\n\t\t\treturn null;\n\t\t} );\n\n\t\tthis.submitButton.onclick = ( event ) => {\n\t\t\tevent.preventDefault();\n\t\t\tthis.saveActiveTable();\n\t\t\treturn false;\n\t\t};\n\t}\n\n\t/**\n\t * Edit label.\n\t *\n\t * @param {Element} label Label to edit.\n\t */\n\teditLabel( label ) {\n\t\tlabel.parentNode.appendChild( this.editLabelInput );\n\t\tthis.editLabelInput.value = label.innerHTML;\n\n\t\tthis.editLabelInput.classList.remove( this.EDIT_LABEL_ERROR_CLASS );\n\t\tthis.editLabelInput.style.display = 'block';\n\t\tthis.editLabelInput.focus();\n\t}\n\n\t/**\n\t * Is new value of edited label unique in active table.\n\t *\n\t * @param {string} newValue New Value from edited label.\n\t * @return {*} If new value of edited label is unique in active table.\n\t */\n\tisUniqueLabel( newValue ) {\n\t\treturn [ ...this.getActiveTable().querySelectorAll( 'label' ) ].reduce(\n\t\t\t( acc, label ) => {\n\t\t\t\treturn acc && label.innerHTML !== newValue;\n\t\t\t},\n\t\t\ttrue\n\t\t);\n\t}\n\n\t/**\n\t * Save modified label.\n\t *\n\t * @param {boolean} cancel Cancel.\n\t */\n\tsaveLabel( cancel = false ) {\n\t\tif ( 'none' === this.editLabelInput.style.display ) {\n\t\t\treturn;\n\t\t}\n\n\t\tconst newValue = this.editLabelInput.value.trim();\n\n\t\tif ( '' === newValue ) {\n\t\t\tconst editedCell = document.getElementById( this.EDIT_LABEL_ID )\n\t\t\t\t.parentElement;\n\t\t\tthis.hideEditLabelInput();\n\t\t\teditedCell.parentNode.removeChild( editedCell );\n\t\t\tthis.setSubmitStatus();\n\n\t\t\treturn;\n\t\t}\n\n\t\tconst label = this.editLabelInput.parentNode.querySelector( 'label' );\n\t\tif ( cancel || newValue === label.innerHTML ) {\n\t\t\tthis.hideEditLabelInput();\n\n\t\t\treturn;\n\t\t}\n\n\t\tif ( ! this.isUniqueLabel( newValue ) ) {\n\t\t\tthis.editLabelInput.classList.add( this.EDIT_LABEL_ERROR_CLASS );\n\n\t\t\treturn;\n\t\t}\n\n\t\tconst input = this.editLabelInput.parentNode.querySelector( 'input' );\n\n\t\tthis.hideEditLabelInput();\n\n\t\tlabel.innerHTML = newValue;\n\t\tthis.replaceName( input, newValue );\n\n\t\tthis.setSubmitStatus();\n\t}\n\n\t/**\n\t * Replace input name according to the new label value.\n\t *\n\t * @param {Element} input Input\n\t * @param {string} newValue New label value\n\t */\n\treplaceName( input, newValue ) {\n\t\t// noinspection JSUnresolvedVariable\n\t\tinput.name = input.name.replace(\n\t\t\t/(.+\\[.+])\\[.*]/g,\n\t\t\t'$1[' + newValue + ']'\n\t\t);\n\t}\n\n\t/**\n\t * Clear message.\n\t *\n\t * @param {HTMLDivElement} message Message.\n\t */\n\tclearMessage( message ) {\n\t\tmessage.innerHTML = '';\n\t\tmessage.classList.remove( 'active' );\n\t}\n\n\t/**\n\t * Clear messages.\n\t */\n\tclearMessages() {\n\t\tthis.clearMessage( this.successMessage );\n\t\tthis.clearMessage( this.errorMessage );\n\t\tclearTimeout( this.msgTimer );\n\t}\n\n\t/**\n\t * Show message.\n\t *\n\t * @param {HTMLDivElement} el Element.\n\t * @param {string} message Message.\n\t */\n\tshowMessage( el, message ) {\n\t\tel.innerHTML = message;\n\t\tel.classList.add( 'active' );\n\n\t\tthis.msgTimer = setTimeout( () => {\n\t\t\tthis.clearMessages();\n\t\t}, 5000 );\n\t}\n}\n\nexport default Tables;\n"],"sourceRoot":""}
classes/class-settings.php DELETED
@@ -1,785 +0,0 @@
1
- <?php
2
- /**
3
- * Plugin Settings.
4
- *
5
- * @package cyr-to-lat
6
- */
7
-
8
- namespace Cyr_To_Lat;
9
-
10
- use Cyr_To_Lat\Symfony\Polyfill\Mbstring\Mbstring;
11
-
12
- /**
13
- * Class Settings
14
- *
15
- * @class Settings
16
- */
17
- class Settings {
18
-
19
- /**
20
- * Admin screen id.
21
- */
22
- const SCREEN_ID = 'settings_page_cyr-to-lat';
23
-
24
- /**
25
- * Option group.
26
- */
27
- const OPTION_GROUP = 'cyr_to_lat_group';
28
-
29
- /**
30
- * Option page.
31
- */
32
- const PAGE = 'cyr-to-lat';
33
-
34
- /**
35
- * Plugin options name.
36
- */
37
- const OPTION_NAME = 'cyr_to_lat_settings';
38
-
39
- /**
40
- * Script handle.
41
- */
42
- const HANDLE = 'cyr-to-lat-settings';
43
-
44
- /**
45
- * Script localization object.
46
- */
47
- const OBJECT = 'Cyr2LatSettingsObject';
48
-
49
- /**
50
- * Form fields.
51
- *
52
- * @var array
53
- */
54
- public $form_fields;
55
-
56
- /**
57
- * Plugin options.
58
- *
59
- * @var array
60
- */
61
- public $settings;
62
-
63
- /**
64
- * Served locales.
65
- *
66
- * @var array
67
- */
68
- protected $locales = [];
69
-
70
- /**
71
- * Settings constructor.
72
- */
73
- public function __construct() {
74
- add_action( 'plugins_loaded', [ $this, 'init' ] );
75
- }
76
-
77
- /**
78
- * Init plugin.
79
- */
80
- public function init() {
81
- $this->load_plugin_textdomain();
82
- $this->init_form_fields();
83
- $this->init_settings();
84
- $this->init_hooks();
85
- }
86
-
87
- /**
88
- * Init class hooks.
89
- */
90
- public function init_hooks() {
91
- add_filter(
92
- 'plugin_action_links_' . plugin_basename( constant( 'CYR_TO_LAT_FILE' ) ),
93
- [ $this, 'add_settings_link' ],
94
- 10,
95
- 4
96
- );
97
-
98
- add_action( 'admin_menu', [ $this, 'add_settings_page' ] );
99
- add_action( 'current_screen', [ $this, 'setup_sections' ] );
100
- add_action( 'current_screen', [ $this, 'setup_fields' ] );
101
-
102
- add_filter( 'pre_update_option_' . self::OPTION_NAME, [ $this, 'pre_update_option_filter' ], 10, 3 );
103
-
104
- add_action( 'admin_enqueue_scripts', [ $this, 'admin_enqueue_scripts' ] );
105
- add_action( 'in_admin_header', [ $this, 'in_admin_header' ] );
106
- }
107
-
108
- /**
109
- * Add link to plugin setting page on plugins page.
110
- *
111
- * @param array $actions An array of plugin action links. By default this can include 'activate',
112
- * 'deactivate', and 'delete'. With Multisite active this can also include
113
- * 'network_active' and 'network_only' items.
114
- * @param string $plugin_file Path to the plugin file relative to the plugins directory.
115
- * @param array $plugin_data An array of plugin data. See `get_plugin_data()`.
116
- * @param string $context The plugin context. By default this can include 'all', 'active', 'inactive',
117
- * 'recently_activated', 'upgrade', 'mustuse', 'dropins', and 'search'.
118
- *
119
- * @return array|mixed Plugin links
120
- * @noinspection PhpUnusedParameterInspection
121
- */
122
- public function add_settings_link( $actions, $plugin_file, $plugin_data, $context ) {
123
- $ctl_actions = [
124
- 'settings' =>
125
- '<a href="' . admin_url( 'options-general.php?page=' . self::PAGE ) .
126
- '" aria-label="' . esc_attr__( 'View Cyr To Lat settings', 'cyr2lat' ) . '">' .
127
- esc_html__( 'Settings', 'cyr2lat' ) . '</a>',
128
- ];
129
-
130
- return array_merge( $ctl_actions, $actions );
131
- }
132
-
133
- /**
134
- * Init locales.
135
- */
136
- protected function init_locales() {
137
- if ( ! empty( $this->locales ) ) {
138
- return;
139
- }
140
-
141
- $this->locales = [
142
- 'iso9' => [
143
- 'label' => __( 'ISO9 Table', 'cyr2lat' ),
144
- ],
145
- 'bel' => [
146
- 'label' => __( 'bel Table', 'cyr2lat' ),
147
- ],
148
- 'uk' => [
149
- 'label' => __( 'uk Table', 'cyr2lat' ),
150
- ],
151
- 'bg_BG' => [
152
- 'label' => __( 'bg_BG Table', 'cyr2lat' ),
153
- ],
154
- 'mk_MK' => [
155
- 'label' => __( 'mk_MK Table', 'cyr2lat' ),
156
- ],
157
- 'sr_RS' => [
158
- 'label' => __( 'sr_RS Table', 'cyr2lat' ),
159
- ],
160
- 'el' => [
161
- 'label' => __( 'el Table', 'cyr2lat' ),
162
- ],
163
- 'hy' => [
164
- 'label' => __( 'hy Table', 'cyr2lat' ),
165
- ],
166
- 'ka_GE' => [
167
- 'label' => __( 'ka_GE Table', 'cyr2lat' ),
168
- ],
169
- 'kk' => [
170
- 'label' => __( 'kk Table', 'cyr2lat' ),
171
- ],
172
- 'he_IL' => [
173
- 'label' => __( 'he_IL Table', 'cyr2lat' ),
174
- ],
175
- 'zh_CN' => [
176
- 'label' => __( 'zh_CN Table', 'cyr2lat' ),
177
- ],
178
- ];
179
- }
180
-
181
- /**
182
- * Get current locale.
183
- *
184
- * @return string
185
- */
186
- private function get_current_locale() {
187
- $current_locale = get_locale();
188
-
189
- return array_key_exists( $current_locale, $this->locales ) ? $current_locale : 'iso9';
190
- }
191
-
192
- /**
193
- * Init options form fields.
194
- */
195
- public function init_form_fields() {
196
- $this->init_locales();
197
-
198
- $current_locale = $this->get_current_locale();
199
-
200
- $this->form_fields = [];
201
-
202
- foreach ( $this->locales as $locale => $info ) {
203
- $current = ( $locale === $current_locale ) ? '<br>' . __( '(current)', 'cyr2lat' ) : '';
204
-
205
- $this->form_fields[ $locale ] = [
206
- 'label' => $info['label'] . $current,
207
- 'section' => $locale . '_section',
208
- 'type' => 'table',
209
- 'placeholder' => '',
210
- 'helper' => '',
211
- 'supplemental' => '',
212
- 'default' => Conversion_Tables::get( $locale ),
213
- ];
214
- }
215
- }
216
-
217
- /**
218
- * Initialise Settings.
219
- *
220
- * Store all settings in a single database entry
221
- * and make sure the $settings array is either the default
222
- * or the settings stored in the database.
223
- */
224
- public function init_settings() {
225
- $this->settings = get_option( self::OPTION_NAME, null );
226
-
227
- $form_fields = $this->get_form_fields();
228
-
229
- // If there are no settings defined, use defaults.
230
- if ( ! is_array( $this->settings ) ) {
231
- $this->settings = array_merge( array_fill_keys( array_keys( $form_fields ), '' ), wp_list_pluck( $form_fields, 'default' ) );
232
- } else {
233
- $this->settings = array_merge( wp_list_pluck( $form_fields, 'default' ), $this->settings );
234
- }
235
- }
236
-
237
- /**
238
- * Get the form fields after they are initialized.
239
- *
240
- * @return array of options
241
- */
242
- public function get_form_fields() {
243
- if ( empty( $this->form_fields ) ) {
244
- $this->init_form_fields();
245
- }
246
-
247
- return array_map( [ $this, 'set_defaults' ], $this->form_fields );
248
- }
249
-
250
- /**
251
- * Set default required properties for each field.
252
- *
253
- * @param array $field Settings field.
254
- *
255
- * @return array
256
- */
257
- protected function set_defaults( $field ) {
258
- if ( ! isset( $field['default'] ) ) {
259
- $field['default'] = '';
260
- }
261
-
262
- return $field;
263
- }
264
-
265
- /**
266
- * Add settings page to the menu.
267
- */
268
- public function add_settings_page() {
269
- $parent_slug = 'options-general.php';
270
- $page_title = __( 'Cyr To Lat', 'cyr2lat' );
271
- $menu_title = __( 'Cyr To Lat', 'cyr2lat' );
272
- $capability = 'manage_options';
273
- $slug = self::PAGE;
274
- $callback = [ $this, 'settings_page' ];
275
- add_submenu_page( $parent_slug, $page_title, $menu_title, $capability, $slug, $callback );
276
- }
277
-
278
- /**
279
- * Settings page.
280
- */
281
- public function settings_page() {
282
- if ( ! $this->is_options_screen() ) {
283
- return;
284
- }
285
-
286
- ?>
287
- <div class="wrap">
288
- <h2 id="title">
289
- <?php
290
- // Admin panel title.
291
- echo( esc_html( __( 'Cyr To Lat Plugin Options', 'cyr2lat' ) ) );
292
- ?>
293
- </h2>
294
-
295
- <form id="ctl-options" action="<?php echo esc_url( admin_url( 'options.php' ) ); ?>" method="post">
296
- <?php
297
- do_settings_sections( self::PAGE ); // Sections with options.
298
- settings_fields( self::OPTION_GROUP ); // Hidden protection fields.
299
- submit_button();
300
- ?>
301
- </form>
302
-
303
- <form id="ctl-convert-existing-slugs" action="" method="post">
304
- <input type="hidden" name="ctl-convert" />
305
- <?php
306
- wp_nonce_field( self::OPTION_GROUP . '-options' );
307
- submit_button( __( 'Convert Existing Slugs', 'cyr2lat' ), 'secondary', 'ctl-convert-button' );
308
- ?>
309
- </form>
310
-
311
- <div id="appreciation">
312
- <h2>
313
- <?php echo esc_html( __( 'Your appreciation', 'cyr2lat' ) ); ?>
314
- </h2>
315
- <a
316
- target="_blank"
317
- href="https://wordpress.org/support/view/plugin-reviews/cyr2lat?rate=5#postform">
318
- <?php echo esc_html( __( 'Leave a ★★★★★ plugin review on WordPress.org', 'cyr2lat' ) ); ?>
319
- </a>
320
- </div>
321
- </div>
322
- <?php
323
- }
324
-
325
- /**
326
- * Setup settings sections.
327
- */
328
- public function setup_sections() {
329
- if ( ! $this->is_options_screen() ) {
330
- return;
331
- }
332
-
333
- foreach ( $this->form_fields as $form_field ) {
334
- add_settings_section(
335
- $form_field['section'],
336
- $form_field['label'],
337
- [ $this, 'section_callback' ],
338
- self::PAGE
339
- );
340
- }
341
- }
342
-
343
- /**
344
- * Section callback.
345
- *
346
- * @param array $arguments Section arguments.
347
- */
348
- public function section_callback( $arguments ) {
349
- $locale = str_replace( '_section', '', $arguments['id'] );
350
- if ( $this->get_current_locale() === $locale ) {
351
- echo '<div id="ctl-current"></div>';
352
- }
353
- }
354
-
355
- /**
356
- * Setup settings fields.
357
- */
358
- public function setup_fields() {
359
- if ( ! $this->is_options_screen() ) {
360
- return;
361
- }
362
-
363
- register_setting( self::OPTION_GROUP, self::OPTION_NAME );
364
-
365
- foreach ( $this->form_fields as $key => $field ) {
366
- $field['field_id'] = $key;
367
-
368
- add_settings_field(
369
- $key,
370
- $field['label'],
371
- [ $this, 'field_callback' ],
372
- self::PAGE,
373
- $field['section'],
374
- $field
375
- );
376
- }
377
- }
378
-
379
- /**
380
- * Output settings field.
381
- *
382
- * @param array $arguments Field arguments.
383
- */
384
- public function field_callback( $arguments ) {
385
- if ( ! isset( $arguments['field_id'] ) ) {
386
- return;
387
- }
388
-
389
- $value = $this->get_option( $arguments['field_id'] );
390
-
391
- // Check which type of field we want.
392
- switch ( $arguments['type'] ) {
393
- case 'text':
394
- case 'password':
395
- case 'number':
396
- printf(
397
- '<input name="%1$s[%2$s]" id="%2$s" type="%3$s" placeholder="%4$s" value="%5$s" class="regular-text" />',
398
- esc_html( self::OPTION_NAME ),
399
- esc_attr( $arguments['field_id'] ),
400
- esc_attr( $arguments['type'] ),
401
- esc_attr( $arguments['placeholder'] ),
402
- esc_html( $value )
403
- );
404
- break;
405
- case 'textarea':
406
- printf(
407
- '<textarea name="%1$s[%2$s]" id="%2$s" placeholder="%3$s" rows="5" cols="50">%4$s</textarea>',
408
- esc_html( self::OPTION_NAME ),
409
- esc_attr( $arguments['field_id'] ),
410
- esc_attr( $arguments['placeholder'] ),
411
- wp_kses_post( $value )
412
- );
413
- break;
414
- case 'checkbox':
415
- case 'radio':
416
- if ( 'checkbox' === $arguments['type'] ) {
417
- $arguments['options'] = [ 'yes' => '' ];
418
- }
419
-
420
- if ( ! empty( $arguments['options'] ) && is_array( $arguments['options'] ) ) {
421
- $options_markup = '';
422
- $iterator = 0;
423
- foreach ( $arguments['options'] as $key => $label ) {
424
- $iterator ++;
425
- $options_markup .= sprintf(
426
- '<label for="%2$s_%7$s"><input id="%2$s_%7$s" name="%1$s[%2$s]" type="%3$s" value="%4$s" %5$s /> %6$s</label><br/>',
427
- esc_html( self::OPTION_NAME ),
428
- $arguments['field_id'],
429
- $arguments['type'],
430
- $key,
431
- checked( $value, $key, false ),
432
- $label,
433
- $iterator
434
- );
435
- }
436
- printf(
437
- '<fieldset>%s</fieldset>',
438
- wp_kses(
439
- $options_markup,
440
- [
441
- 'label' => [
442
- 'for' => [],
443
- ],
444
- 'input' => [
445
- 'id' => [],
446
- 'name' => [],
447
- 'type' => [],
448
- 'value' => [],
449
- 'checked' => [],
450
- ],
451
- 'br' => [],
452
- ]
453
- )
454
- );
455
- }
456
- break;
457
- case 'select': // If it is a select dropdown.
458
- if ( ! empty( $arguments['options'] ) && is_array( $arguments['options'] ) ) {
459
- $options_markup = '';
460
- foreach ( $arguments['options'] as $key => $label ) {
461
- $options_markup .= sprintf(
462
- '<option value="%s" %s>%s</option>',
463
- $key,
464
- selected( $value, $key, false ),
465
- $label
466
- );
467
- }
468
- printf(
469
- '<select name="%1$s[%2$s]">%3$s</select>',
470
- esc_html( self::OPTION_NAME ),
471
- esc_html( $arguments['field_id'] ),
472
- wp_kses(
473
- $options_markup,
474
- [
475
- 'option' => [
476
- 'value' => [],
477
- 'selected' => [],
478
- ],
479
- ]
480
- )
481
- );
482
- }
483
- break;
484
- case 'multiple': // If it is a multiple select dropdown.
485
- if ( ! empty( $arguments['options'] ) && is_array( $arguments['options'] ) ) {
486
- $options_markup = '';
487
- foreach ( $arguments['options'] as $key => $label ) {
488
- $selected = '';
489
- if ( is_array( $value ) && in_array( $key, $value, true ) ) {
490
- $selected = selected( $key, $key, false );
491
- }
492
- $options_markup .= sprintf(
493
- '<option value="%s" %s>%s</option>',
494
- $key,
495
- $selected,
496
- $label
497
- );
498
- }
499
- printf(
500
- '<select multiple="multiple" name="%1$s[%2$s][]">%3$s</select>',
501
- esc_html( self::OPTION_NAME ),
502
- esc_html( $arguments['field_id'] ),
503
- wp_kses(
504
- $options_markup,
505
- [
506
- 'option' => [
507
- 'value' => [],
508
- 'selected' => [],
509
- ],
510
- ]
511
- )
512
- );
513
- }
514
- break;
515
- case 'table':
516
- if ( is_array( $value ) ) {
517
- $iterator = 0;
518
- foreach ( $value as $key => $cell_value ) {
519
- $id = $arguments['field_id'] . '-' . $iterator;
520
-
521
- echo '<div class="ctl-table-cell">';
522
- printf(
523
- '<label for="%1$s">%2$s</label>',
524
- esc_html( $id ),
525
- esc_html( $key )
526
- );
527
- printf(
528
- '<input name="%1$s[%2$s][%3$s]" id="%4$s" type="%5$s" placeholder="%6$s" value="%7$s" class="regular-text" />',
529
- esc_html( self::OPTION_NAME ),
530
- esc_attr( $arguments['field_id'] ),
531
- esc_attr( $key ),
532
- esc_attr( $id ),
533
- 'text',
534
- esc_attr( $arguments['placeholder'] ),
535
- esc_html( $cell_value )
536
- );
537
- echo '</div>';
538
-
539
- $iterator ++;
540
- }
541
- }
542
- break;
543
- default:
544
- break;
545
- }
546
-
547
- // If there is help text.
548
- $helper = $arguments['helper'];
549
- if ( $helper ) {
550
- printf( '<span class="helper"> %s</span>', esc_html( $helper ) );
551
- }
552
-
553
- // If there is supplemental text.
554
- $supplemental = $arguments['supplemental'];
555
- if ( $supplemental ) {
556
- printf( '<p class="description">%s</p>', esc_html( $supplemental ) );
557
- }
558
- }
559
-
560
- /**
561
- * Get plugin option.
562
- *
563
- * @param string $key Setting name.
564
- * @param mixed $empty_value Empty value for this setting.
565
- *
566
- * @return string|array The value specified for the option or a default value for the option.
567
- */
568
- public function get_option( $key, $empty_value = null ) {
569
- if ( empty( $this->settings ) ) {
570
- $this->init_settings();
571
- }
572
-
573
- // Get option default if unset.
574
- if ( ! isset( $this->settings[ $key ] ) ) {
575
- $form_fields = $this->get_form_fields();
576
- $this->settings[ $key ] = isset( $form_fields[ $key ] ) ? $this->get_field_default( $form_fields[ $key ] ) : '';
577
- }
578
-
579
- if ( '' === $this->settings[ $key ] && ! is_null( $empty_value ) ) {
580
- $this->settings[ $key ] = $empty_value;
581
- }
582
-
583
- return $this->settings[ $key ];
584
- }
585
-
586
- /**
587
- * Get a field default value. Defaults to '' if not set.
588
- *
589
- * @param array $field Setting field default value.
590
- *
591
- * @return string
592
- */
593
- protected function get_field_default( $field ) {
594
- return empty( $field['default'] ) ? '' : $field['default'];
595
- }
596
-
597
- /**
598
- * Set plugin option.
599
- *
600
- * @param string $key Setting name.
601
- * @param mixed $value Setting value.
602
- */
603
- public function set_option( $key, $value ) {
604
- if ( empty( $this->settings ) ) {
605
- $this->init_settings();
606
- }
607
-
608
- $this->settings[ $key ] = $value;
609
- update_option( self::OPTION_NAME, $this->settings );
610
- }
611
-
612
- /**
613
- * Filter plugin option update.
614
- *
615
- * @param mixed $value New option value.
616
- * @param mixed $old_value Old option value.
617
- * @param string $option Option name.
618
- *
619
- * @return mixed
620
- * @noinspection PhpUnusedParameterInspection
621
- */
622
- public function pre_update_option_filter( $value, $old_value, $option ) {
623
- if ( $value === $old_value ) {
624
- return $value;
625
- }
626
-
627
- // We save only one table, so merge with all existing tables.
628
- if ( is_array( $old_value ) && ( is_array( $value ) ) ) {
629
- $value = array_merge( $old_value, $value );
630
- }
631
-
632
- $form_fields = $this->get_form_fields();
633
- foreach ( $form_fields as $key => $form_field ) {
634
- if ( 'checkbox' === $form_field['type'] ) {
635
- $form_field_value = isset( $value[ $key ] ) ? $value[ $key ] : 'no';
636
- $form_field_value = '1' === $form_field_value || 'yes' === $form_field_value ? 'yes' : 'no';
637
- $value[ $key ] = $form_field_value;
638
- }
639
- }
640
-
641
- return $value;
642
- }
643
-
644
- /**
645
- * Enqueue class scripts.
646
- */
647
- public function admin_enqueue_scripts() {
648
- if ( ! $this->is_options_screen() ) {
649
- return;
650
- }
651
-
652
- wp_enqueue_script(
653
- self::HANDLE,
654
- constant( 'CYR_TO_LAT_URL' ) . '/dist/js/settings/app.js',
655
- [],
656
- constant( 'CYR_TO_LAT_VERSION' ),
657
- true
658
- );
659
-
660
- wp_localize_script(
661
- self::HANDLE,
662
- self::OBJECT,
663
- [
664
- 'optionsSaveSuccessMessage' => __( 'Options saved.', 'cyr2lat' ),
665
- 'optionsSaveErrorMessage' => __( 'Error saving options.', 'cyr2lat' ),
666
- ]
667
- );
668
-
669
- wp_enqueue_style(
670
- self::HANDLE,
671
- constant( 'CYR_TO_LAT_URL' ) . '/css/cyr-to-lat-admin.css',
672
- [],
673
- constant( 'CYR_TO_LAT_VERSION' )
674
- );
675
- }
676
-
677
- /**
678
- * Output convert confirmation popup.
679
- */
680
- public function in_admin_header() {
681
- if ( ! $this->is_options_screen() ) {
682
- return;
683
- }
684
-
685
- ?>
686
- <div id="ctl-confirm-popup">
687
- <div id="ctl-confirm-content">
688
- <p>
689
- <strong><?php esc_html_e( 'Important:', 'cyr2lat' ); ?></strong>
690
- <?php
691
- esc_html_e(
692
- 'This operation is irreversible. Please make sure that you have made backup copy of your database.',
693
- 'cyr2lat'
694
- );
695
- ?>
696
- </p>
697
- <p><?php esc_html_e( 'Are you sure to continue?', 'cyr2lat' ); ?></p>
698
- <div id="ctl-confirm-buttons">
699
- <input
700
- type="button" id="ctl-confirm-ok" class="button button-primary"
701
- value="<?php esc_html_e( 'OK', 'cyr2lat' ); ?>">
702
- <button
703
- type="button" id="ctl-confirm-cancel" class="button button-secondary">
704
- <?php esc_html_e( 'Cancel', 'cyr2lat' ); ?>
705
- </button>
706
- </div>
707
- </div>
708
- </div>
709
- <?php
710
- }
711
-
712
- /**
713
- * Load plugin text domain.
714
- */
715
- public function load_plugin_textdomain() {
716
- load_plugin_textdomain(
717
- 'cyr2lat',
718
- false,
719
- dirname( plugin_basename( constant( 'CYR_TO_LAT_FILE' ) ) ) . '/languages/'
720
- );
721
- }
722
-
723
- /**
724
- * Get transliteration table.
725
- *
726
- * @return array
727
- */
728
- public function get_table() {
729
- // List of locales: https://make.wordpress.org/polyglots/teams/.
730
- $locale = get_locale();
731
- $table = $this->get_option( $locale );
732
- if ( empty( $table ) ) {
733
- $table = $this->get_option( 'iso9' );
734
- }
735
-
736
- return $this->transpose_chinese_table( $table );
737
- }
738
-
739
- /**
740
- * Is current locale a Chinese one.
741
- *
742
- * @return bool
743
- */
744
- public function is_chinese_locale() {
745
- $chinese_locales = [ 'zh_CN', 'zh_HK', 'zh_SG', 'zh_TW' ];
746
-
747
- return in_array( get_locale(), $chinese_locales, true );
748
- }
749
-
750
- /**
751
- * Transpose Chinese table.
752
- *
753
- * Chinese tables are stored in different way, to show them compact.
754
- *
755
- * @param array $table Table.
756
- *
757
- * @return array
758
- */
759
- protected function transpose_chinese_table( $table ) {
760
- if ( ! $this->is_chinese_locale() ) {
761
- return $table;
762
- }
763
-
764
- $transposed_table = [];
765
- foreach ( $table as $key => $item ) {
766
- $hieroglyphs = Mbstring::mb_str_split( $item );
767
- foreach ( $hieroglyphs as $hieroglyph ) {
768
- $transposed_table[ $hieroglyph ] = $key;
769
- }
770
- }
771
-
772
- return $transposed_table;
773
- }
774
-
775
- /**
776
- * Is current admin screen the plugin options screen.
777
- *
778
- * @return bool
779
- */
780
- protected function is_options_screen() {
781
- $current_screen = get_current_screen();
782
-
783
- return $current_screen && ( 'options' === $current_screen->id || self::SCREEN_ID === $current_screen->id );
784
- }
785
- }
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
cyr-to-lat.php CHANGED
@@ -2,21 +2,23 @@
2
  /**
3
  * Cyr-To-Lat
4
  *
5
- * Plugin Name: Cyr-To-Lat
6
- * Plugin URI: https://wordpress.org/plugins/cyr2lat/
7
- * Description: Converts Cyrillic characters in post and term slugs to Latin characters. Useful for creating human-readable URLs. Based on the original plugin by Anton Skorobogatov.
8
- * Author: Sergey Biryukov, Mikhail Kobzarev, Igor Gergel
9
- * Author URI: https://profiles.wordpress.org/sergeybiryukov/
10
- * Requires at least: 5.1
11
- * Tested up to: 5.7
12
- * Version: 4.6.4
13
- * Stable tag: 4.6.4
14
- *
15
- * Text Domain: cyr2lat
16
- * Domain Path: /languages/
17
  *
18
- * @package cyr-to-lat
19
- * @author Sergey Biryukov, Mikhail Kobzarev, Igor Gergel
 
 
 
 
 
 
 
 
 
 
20
  */
21
 
22
  namespace Cyr_To_Lat;
@@ -34,7 +36,7 @@ if ( defined( 'CYR_TO_LAT_VERSION' ) ) {
34
  /**
35
  * Plugin version.
36
  */
37
- define( 'CYR_TO_LAT_VERSION', '4.6.4' );
38
 
39
  /**
40
  * Path to the plugin dir.
@@ -79,14 +81,9 @@ define( 'CYR_TO_LAT_REQUIRED_MAX_INPUT_VARS', 1000 );
79
  /**
80
  * Init plugin on plugin load.
81
  */
82
- require_once constant( 'CYR_TO_LAT_PATH' ) . '/vendor/autoload.php';
83
-
84
- $cyr_to_lat_requirements = new Requirements();
85
-
86
- if ( ! $cyr_to_lat_requirements->are_requirements_met() ) {
87
- return;
88
- }
89
 
90
  global $cyr_to_lat_plugin;
91
 
92
  $cyr_to_lat_plugin = new Main();
 
2
  /**
3
  * Cyr-To-Lat
4
  *
5
+ * @package cyr-to-lat
6
+ * @author Sergey Biryukov, Mikhail Kobzarev, Igor Gergel
7
+ * @license GPL-2.0-or-later
8
+ * @wordpress-plugin
 
 
 
 
 
 
 
 
9
  *
10
+ * Plugin Name: Cyr-To-Lat
11
+ * Plugin URI: https://wordpress.org/plugins/cyr2lat/
12
+ * Description: Converts Cyrillic characters in post and term slugs to Latin characters. Useful for creating human-readable URLs. Based on the original plugin by Anton Skorobogatov.
13
+ * Version: 5.0.0
14
+ * Requires at least: 5.1
15
+ * Requires PHP: 5.6.20
16
+ * Author: Sergey Biryukov, Mikhail Kobzarev, Igor Gergel
17
+ * Author URI: https://profiles.wordpress.org/sergeybiryukov/
18
+ * License: GPL v2 or later
19
+ * License URI: https://www.gnu.org/licenses/gpl-2.0.html
20
+ * Text Domain: cyr2lat
21
+ * Domain Path: /languages/
22
  */
23
 
24
  namespace Cyr_To_Lat;
36
  /**
37
  * Plugin version.
38
  */
39
+ define( 'CYR_TO_LAT_VERSION', '5.0.0' );
40
 
41
  /**
42
  * Path to the plugin dir.
81
  /**
82
  * Init plugin on plugin load.
83
  */
84
+ require_once 'vendor/autoload.php';
 
 
 
 
 
 
85
 
86
  global $cyr_to_lat_plugin;
87
 
88
  $cyr_to_lat_plugin = new Main();
89
+ $cyr_to_lat_plugin->init();
dist/js/settings/app.js DELETED
@@ -1,2 +0,0 @@
1
- !function(t){var e={};function n(r){if(e[r])return e[r].exports;var i=e[r]={i:r,l:!1,exports:{}};return t[r].call(i.exports,i,i.exports,n),i.l=!0,i.exports}n.m=t,n.c=e,n.d=function(t,e,r){n.o(t,e)||Object.defineProperty(t,e,{enumerable:!0,get:r})},n.r=function(t){"undefined"!=typeof Symbol&&Symbol.toStringTag&&Object.defineProperty(t,Symbol.toStringTag,{value:"Module"}),Object.defineProperty(t,"__esModule",{value:!0})},n.t=function(t,e){if(1&e&&(t=n(t)),8&e)return t;if(4&e&&"object"==typeof t&&t&&t.__esModule)return t;var r=Object.create(null);if(n.r(r),Object.defineProperty(r,"default",{enumerable:!0,value:t}),2&e&&"string"!=typeof t)for(var i in t)n.d(r,i,function(e){return t[e]}.bind(null,i));return r},n.n=function(t){var e=t&&t.__esModule?function(){return t.default}:function(){return t};return n.d(e,"a",e),e},n.o=function(t,e){return Object.prototype.hasOwnProperty.call(t,e)},n.p="",n(n.s=0)}([function(t,e,n){n(1),t.exports=n(2)},function(t,e){var n=function(t){function e(){this.fetch=!1,this.DOMException=t.DOMException}return e.prototype=t,new e}("undefined"!=typeof self?self:this);!function(t){!function(e){var n="URLSearchParams"in t,r="Symbol"in t&&"iterator"in Symbol,i="FileReader"in t&&"Blob"in t&&function(){try{return new Blob,!0}catch(t){return!1}}(),o="FormData"in t,s="ArrayBuffer"in t;if(s)var a=["[object Int8Array]","[object Uint8Array]","[object Uint8ClampedArray]","[object Int16Array]","[object Uint16Array]","[object Int32Array]","[object Uint32Array]","[object Float32Array]","[object Float64Array]"],u=ArrayBuffer.isView||function(t){return t&&a.indexOf(Object.prototype.toString.call(t))>-1};function l(t){if("string"!=typeof t&&(t=String(t)),/[^a-z0-9\-#$%&'*+.^_`|~]/i.test(t))throw new TypeError("Invalid character in header field name");return t.toLowerCase()}function c(t){return"string"!=typeof t&&(t=String(t)),t}function h(t){var e={next:function(){var e=t.shift();return{done:void 0===e,value:e}}};return r&&(e[Symbol.iterator]=function(){return e}),e}function d(t){this.map={},t instanceof d?t.forEach((function(t,e){this.append(e,t)}),this):Array.isArray(t)?t.forEach((function(t){this.append(t[0],t[1])}),this):t&&Object.getOwnPropertyNames(t).forEach((function(e){this.append(e,t[e])}),this)}function p(t){if(t.bodyUsed)return Promise.reject(new TypeError("Already read"));t.bodyUsed=!0}function f(t){return new Promise((function(e,n){t.onload=function(){e(t.result)},t.onerror=function(){n(t.error)}}))}function y(t){var e=new FileReader,n=f(e);return e.readAsArrayBuffer(t),n}function b(t){if(t.slice)return t.slice(0);var e=new Uint8Array(t.byteLength);return e.set(new Uint8Array(t)),e.buffer}function v(){return this.bodyUsed=!1,this._initBody=function(t){var e;this._bodyInit=t,t?"string"==typeof t?this._bodyText=t:i&&Blob.prototype.isPrototypeOf(t)?this._bodyBlob=t:o&&FormData.prototype.isPrototypeOf(t)?this._bodyFormData=t:n&&URLSearchParams.prototype.isPrototypeOf(t)?this._bodyText=t.toString():s&&i&&((e=t)&&DataView.prototype.isPrototypeOf(e))?(this._bodyArrayBuffer=b(t.buffer),this._bodyInit=new Blob([this._bodyArrayBuffer])):s&&(ArrayBuffer.prototype.isPrototypeOf(t)||u(t))?this._bodyArrayBuffer=b(t):this._bodyText=t=Object.prototype.toString.call(t):this._bodyText="",this.headers.get("content-type")||("string"==typeof t?this.headers.set("content-type","text/plain;charset=UTF-8"):this._bodyBlob&&this._bodyBlob.type?this.headers.set("content-type",this._bodyBlob.type):n&&URLSearchParams.prototype.isPrototypeOf(t)&&this.headers.set("content-type","application/x-www-form-urlencoded;charset=UTF-8"))},i&&(this.blob=function(){var t=p(this);if(t)return t;if(this._bodyBlob)return Promise.resolve(this._bodyBlob);if(this._bodyArrayBuffer)return Promise.resolve(new Blob([this._bodyArrayBuffer]));if(this._bodyFormData)throw new Error("could not read FormData body as blob");return Promise.resolve(new Blob([this._bodyText]))},this.arrayBuffer=function(){return this._bodyArrayBuffer?p(this)||Promise.resolve(this._bodyArrayBuffer):this.blob().then(y)}),this.text=function(){var t,e,n,r=p(this);if(r)return r;if(this._bodyBlob)return t=this._bodyBlob,e=new FileReader,n=f(e),e.readAsText(t),n;if(this._bodyArrayBuffer)return Promise.resolve(function(t){for(var e=new Uint8Array(t),n=new Array(e.length),r=0;r<e.length;r++)n[r]=String.fromCharCode(e[r]);return n.join("")}(this._bodyArrayBuffer));if(this._bodyFormData)throw new Error("could not read FormData body as text");return Promise.resolve(this._bodyText)},o&&(this.formData=function(){return this.text().then(S)}),this.json=function(){return this.text().then(JSON.parse)},this}d.prototype.append=function(t,e){t=l(t),e=c(e);var n=this.map[t];this.map[t]=n?n+", "+e:e},d.prototype.delete=function(t){delete this.map[l(t)]},d.prototype.get=function(t){return t=l(t),this.has(t)?this.map[t]:null},d.prototype.has=function(t){return this.map.hasOwnProperty(l(t))},d.prototype.set=function(t,e){this.map[l(t)]=c(e)},d.prototype.forEach=function(t,e){for(var n in this.map)this.map.hasOwnProperty(n)&&t.call(e,this.map[n],n,this)},d.prototype.keys=function(){var t=[];return this.forEach((function(e,n){t.push(n)})),h(t)},d.prototype.values=function(){var t=[];return this.forEach((function(e){t.push(e)})),h(t)},d.prototype.entries=function(){var t=[];return this.forEach((function(e,n){t.push([n,e])})),h(t)},r&&(d.prototype[Symbol.iterator]=d.prototype.entries);var m=["DELETE","GET","HEAD","OPTIONS","POST","PUT"];function E(t,e){var n,r,i=(e=e||{}).body;if(t instanceof E){if(t.bodyUsed)throw new TypeError("Already read");this.url=t.url,this.credentials=t.credentials,e.headers||(this.headers=new d(t.headers)),this.method=t.method,this.mode=t.mode,this.signal=t.signal,i||null==t._bodyInit||(i=t._bodyInit,t.bodyUsed=!0)}else this.url=String(t);if(this.credentials=e.credentials||this.credentials||"same-origin",!e.headers&&this.headers||(this.headers=new d(e.headers)),this.method=(n=e.method||this.method||"GET",r=n.toUpperCase(),m.indexOf(r)>-1?r:n),this.mode=e.mode||this.mode||null,this.signal=e.signal||this.signal,this.referrer=null,("GET"===this.method||"HEAD"===this.method)&&i)throw new TypeError("Body not allowed for GET or HEAD requests");this._initBody(i)}function S(t){var e=new FormData;return t.trim().split("&").forEach((function(t){if(t){var n=t.split("="),r=n.shift().replace(/\+/g," "),i=n.join("=").replace(/\+/g," ");e.append(decodeURIComponent(r),decodeURIComponent(i))}})),e}function L(t,e){e||(e={}),this.type="default",this.status=void 0===e.status?200:e.status,this.ok=this.status>=200&&this.status<300,this.statusText="statusText"in e?e.statusText:"OK",this.headers=new d(e.headers),this.url=e.url||"",this._initBody(t)}E.prototype.clone=function(){return new E(this,{body:this._bodyInit})},v.call(E.prototype),v.call(L.prototype),L.prototype.clone=function(){return new L(this._bodyInit,{status:this.status,statusText:this.statusText,headers:new d(this.headers),url:this.url})},L.error=function(){var t=new L(null,{status:0,statusText:""});return t.type="error",t};var T=[301,302,303,307,308];L.redirect=function(t,e){if(-1===T.indexOf(e))throw new RangeError("Invalid status code");return new L(null,{status:e,headers:{location:t}})},e.DOMException=t.DOMException;try{new e.DOMException}catch(t){e.DOMException=function(t,e){this.message=t,this.name=e;var n=Error(t);this.stack=n.stack},e.DOMException.prototype=Object.create(Error.prototype),e.DOMException.prototype.constructor=e.DOMException}function _(t,n){return new Promise((function(r,o){var s=new E(t,n);if(s.signal&&s.signal.aborted)return o(new e.DOMException("Aborted","AbortError"));var a=new XMLHttpRequest;function u(){a.abort()}a.onload=function(){var t,e,n={status:a.status,statusText:a.statusText,headers:(t=a.getAllResponseHeaders()||"",e=new d,t.replace(/\r?\n[\t ]+/g," ").split(/\r?\n/).forEach((function(t){var n=t.split(":"),r=n.shift().trim();if(r){var i=n.join(":").trim();e.append(r,i)}})),e)};n.url="responseURL"in a?a.responseURL:n.headers.get("X-Request-URL");var i="response"in a?a.response:a.responseText;r(new L(i,n))},a.onerror=function(){o(new TypeError("Network request failed"))},a.ontimeout=function(){o(new TypeError("Network request failed"))},a.onabort=function(){o(new e.DOMException("Aborted","AbortError"))},a.open(s.method,s.url,!0),"include"===s.credentials?a.withCredentials=!0:"omit"===s.credentials&&(a.withCredentials=!1),"responseType"in a&&i&&(a.responseType="blob"),s.headers.forEach((function(t,e){a.setRequestHeader(e,t)})),s.signal&&(s.signal.addEventListener("abort",u),a.onreadystatechange=function(){4===a.readyState&&s.signal.removeEventListener("abort",u)}),a.send(void 0===s._bodyInit?null:s._bodyInit)}))}_.polyfill=!0,t.fetch||(t.fetch=_,t.Headers=d,t.Request=E,t.Response=L),e.Headers=d,e.Request=E,e.Response=L,e.fetch=_}({})}(n),delete n.fetch.polyfill,(e=n.fetch).default=n.fetch,e.fetch=n.fetch,e.Headers=n.Headers,e.Request=n.Request,e.Response=n.Response,t.exports=e},function(t,e,n){"use strict";var r,i=n(3),o=(r=i)&&r.__esModule?r:{default:r};document.addEventListener("DOMContentLoaded",(function(){new o.default}))},function(t,e,n){"use strict";Object.defineProperty(e,"__esModule",{value:!0});var r=function(){function t(t,e){for(var n=0;n<e.length;n++){var r=e[n];r.enumerable=r.enumerable||!1,r.configurable=!0,"value"in r&&(r.writable=!0),Object.defineProperty(t,r.key,r)}}return function(e,n,r){return n&&t(e.prototype,n),r&&t(e,r),e}}();function i(t){if(Array.isArray(t)){for(var e=0,n=Array(t.length);e<t.length;e++)n[e]=t[e];return n}return Array.from(t)}var o=function(){function t(){!function(t,e){if(!(t instanceof e))throw new TypeError("Cannot call a class as a function")}(this,t),this.OPTIONS_FORM_SELECTOR="#ctl-options",this.HEADER_SELECTOR=this.OPTIONS_FORM_SELECTOR+" h2",this.TABLE_SELECTOR=this.OPTIONS_FORM_SELECTOR+" table",this.SUBMIT_SELECTOR=this.OPTIONS_FORM_SELECTOR+" #submit",this.CONVERT_FORM_SELECTOR="#ctl-convert-existing-slugs",this.CONVERT_BUTTON_SELECTOR="#ctl-convert-button",this.CONFIRM_POPUP_SELECTOR="#ctl-confirm-popup",this.CONFIRM_OK_SELECTOR="#ctl-confirm-ok",this.CONFIRM_CANCEL_SELECTOR="#ctl-confirm-cancel",this.CURRENT_STUB_ID="ctl-current",this.CURRENT_NAV_TAB_CLASS="nav-tab-current",this.ACTIVE_NAV_TAB_CLASS="nav-tab-active",this.ACTIVE_TABLE_CLASS="active",this.EDIT_LABEL_ID="ctl-edit-label",this.EDIT_LABEL_ERROR_CLASS="ctl-edit-label-error",this.plusButton='<button type="button" aria-haspopup="true" aria-expanded="false" class="components-button block-editor-inserter__toggle has-icon" aria-label="Добавить блок"><svg width="24" height="24" xmlns="http://www.w3.org/2000/svg" viewBox="-2 -2 24 24" role="img" aria-hidden="true" focusable="false"><path d="M10 1c-5 0-9 4-9 9s4 9 9 9 9-4 9-9-4-9-9-9zm0 16c-3.9 0-7-3.1-7-7s3.1-7 7-7 7 3.1 7 7-3.1 7-7 7zm1-11H9v3H6v2h3v3h2v-3h3V9h-3V6zM10 1c-5 0-9 4-9 9s4 9 9 9 9-4 9-9-4-9-9-9zm0 16c-3.9 0-7-3.1-7-7s3.1-7 7-7 7 3.1 7 7-3.1 7-7 7zm1-11H9v3H6v2h3v3h2v-3h3V9h-3V6z"></path></svg></button>',this.PLUS_CLASS="ctl-plus",this.optionsForm=document.querySelector(this.OPTIONS_FORM_SELECTOR),this.tablesData=this.getTablesData(),this.submitButton=document.querySelector(this.SUBMIT_SELECTOR),this.confirmPopup=document.querySelector(this.CONFIRM_POPUP_SELECTOR),this.optionsSaveSuccessMessage=Cyr2LatSettingsObject.optionsSaveSuccessMessage,this.optionsSaveErrorMessage=Cyr2LatSettingsObject.optionsSaveErrorMessage,this.addWrapper(),this.addMessageLines(),this.addEditLabelInput(),this.hideTables(),this.bindEvents(),this.setSubmitStatus()}return r(t,[{key:"getHeaders",value:function(){return[].concat(i(document.querySelectorAll(this.HEADER_SELECTOR)))}},{key:"getActiveHeader",value:function(){return document.querySelector(this.HEADER_SELECTOR+"."+this.ACTIVE_NAV_TAB_CLASS)}},{key:"getActiveIndex",value:function(){return this.getActiveHeader().dataset.index}},{key:"getTables",value:function(){return[].concat(i(document.querySelectorAll(this.TABLE_SELECTOR)))}},{key:"getActiveTable",value:function(){return document.querySelector(this.TABLE_SELECTOR+"."+this.ACTIVE_TABLE_CLASS)}},{key:"getInputs",value:function(){return[].concat(i(document.querySelectorAll(this.OPTIONS_FORM_SELECTOR+" input")))}},{key:"getLabels",value:function(){return[].concat(i(document.querySelectorAll(this.OPTIONS_FORM_SELECTOR+" label")))}},{key:"getPlusButtons",value:function(){return[].concat(i(document.querySelectorAll(this.OPTIONS_FORM_SELECTOR+" ."+this.PLUS_CLASS)))}},{key:"isActiveTableChanged",value:function(){var t=this.getActiveIndex();return JSON.stringify(this.getActiveTableData())!==JSON.stringify(this.tablesData[t])}},{key:"setSubmitStatus",value:function(){this.submitButton.disabled=!this.isActiveTableChanged()}},{key:"saveActiveTable",value:function(){var t=this;if(this.isActiveTableChanged()){var e=this.getActiveTable(),n=document.createElement("form");return n.action=this.optionsForm.getAttribute("action"),n.method=this.optionsForm.method,n.appendChild(e.cloneNode(!0)),[].concat(i(e.querySelectorAll("input"))).map((function(t){return n.querySelector("#"+t.id).value=t.value})),[].concat(i(this.optionsForm.querySelectorAll('input[type="hidden"]'))).map((function(t){return n.appendChild(t.cloneNode(!0))})),document.body.appendChild(n),fetch(this.optionsForm.getAttribute("action"),{method:n.method,body:new URLSearchParams([].concat(i(new FormData(n))))}).then((function(e){return e.ok?(t.showMessage(t.successMessage,t.optionsSaveSuccessMessage),t.tablesData=t.getTablesData()):t.showMessage(t.errorMessage,t.optionsSaveErrorMessage),e.json()})).finally((function(){n.parentNode.removeChild(n),t.setSubmitStatus()}))}}},{key:"getTableData",value:function(t){var e=this,n=[].concat(i(t.querySelectorAll("input"))),r={};return n.forEach((function(t){var n=document.querySelector(e.OPTIONS_FORM_SELECTOR+' label[for="'+t.id+'"]');r[n.innerHTML]=t.value})),r}},{key:"getTablesData",value:function(){var t=this;return this.getTables().map((function(e){return t.getTableData(e)}))}},{key:"getActiveTableData",value:function(){return this.getTableData(this.getActiveTable())}},{key:"addWrapper",value:function(){this.wrapper=document.createElement("ul"),this.wrapper.classList.add("nav-tab-wrapper"),this.optionsForm.insertBefore(this.wrapper,this.optionsForm.firstChild)}},{key:"addMessageLine",value:function(t){var e=document.createElement("div");return e.id=t,this.optionsForm.insertBefore(e,this.optionsForm.firstChild),e}},{key:"addMessageLines",value:function(){this.successMessage=this.addMessageLine("ctl-success"),this.errorMessage=this.addMessageLine("ctl-error")}},{key:"addEditLabelInput",value:function(){this.editLabelInput=document.createElement("input"),this.editLabelInput.id=this.EDIT_LABEL_ID,this.editLabelInput.style.display="none",document.body.appendChild(this.editLabelInput)}},{key:"hideEditLabelInput",value:function(){this.editLabelInput.style.display="none",this.editLabelInput.classList.remove(this.EDIT_LABEL_ERROR_CLASS),document.body.appendChild(this.editLabelInput)}},{key:"getLastCell",value:function(){return document.querySelector(this.OPTIONS_FORM_SELECTOR+" ."+this.ACTIVE_TABLE_CLASS+" ."+this.PLUS_CLASS).previousElementSibling}},{key:"addCell",value:function(){var t=this.getLastCell();t.parentElement.insertBefore(t.cloneNode(!0),t.nextElementSibling);var e=(t=this.getLastCell()).querySelector("label"),n=t.querySelector("input"),r=n.id.split("-"),i=r[0]+"-"+(parseInt(r[1])+1);e.htmlFor=i,e.innerHTML="",n.id=i,n.value="",n.setAttribute("value",""),this.replaceName(n,""),this.bindEvents(),this.editLabel(e)}},{key:"hideTables",value:function(){var t=this,e=0;this.getTables().map((function(n,r){n.classList.add("ctl-table"),t.CURRENT_STUB_ID===n.previousElementSibling.id&&(e=r,n.classList.add(t.ACTIVE_TABLE_CLASS));var i=document.createElement("div");return i.classList.add(t.PLUS_CLASS),i.innerHTML=t.plusButton,n.querySelector("td").appendChild(i),null})),this.getHeaders().map((function(n,r){return n.classList.add("nav-tab"),n.dataset.index=r,t.wrapper.appendChild(n),r===e&&(n.classList.add(t.CURRENT_NAV_TAB_CLASS),n.classList.add(t.ACTIVE_NAV_TAB_CLASS)),null}))}},{key:"bindEvents",value:function(){var t=this;this.getHeaders().map((function(e,n,r){return e.onclick=function(e){e.preventDefault();var n=e.target.dataset.index;if(n===t.getActiveIndex())return!1;t.saveActiveTable(),r.map((function(e){return e.classList.remove(t.ACTIVE_NAV_TAB_CLASS)})),r[n].classList.add(t.ACTIVE_NAV_TAB_CLASS);var i=t.getTables();return i.map((function(e){return e.classList.remove(t.ACTIVE_TABLE_CLASS)})),i[n].classList.add(t.ACTIVE_TABLE_CLASS),t.setSubmitStatus(),!1},null})),this.getInputs().map((function(e){return e.oninput=function(){t.setSubmitStatus()},null})),this.getLabels().map((function(e){return e.onclick=function(e){return e.preventDefault(),t.editLabel(e.target),!1},null})),this.editLabelInput.onblur=function(){t.saveLabel()},this.editLabelInput.onkeyup=function(e){"Escape"===e.key&&t.saveLabel(!0),"Enter"===e.key&&t.saveLabel()},this.getPlusButtons().map((function(e){return e.onclick=function(e){return e.preventDefault(),t.addCell(e.target),!1},null})),this.submitButton.onclick=function(e){return e.preventDefault(),t.saveActiveTable(),!1},document.querySelector(this.CONVERT_BUTTON_SELECTOR).onclick=function(e){return e.preventDefault(),t.confirmPopup.style.display="block",!1},this.confirmPopup.onclick=function(){t.hideConfirmPopup()},document.querySelector(this.CONFIRM_OK_SELECTOR).onclick=function(e){e.stopPropagation(),t.hideConfirmPopup(),document.querySelector(t.CONVERT_FORM_SELECTOR).submit()},document.querySelector(this.CONFIRM_CANCEL_SELECTOR).onclick=function(e){e.stopPropagation(),t.hideConfirmPopup()}}},{key:"hideConfirmPopup",value:function(){this.confirmPopup.style.display="none"}},{key:"editLabel",value:function(t){t.parentNode.appendChild(this.editLabelInput),this.editLabelInput.value=t.innerHTML,this.editLabelInput.classList.remove(this.EDIT_LABEL_ERROR_CLASS),this.editLabelInput.style.display="block",this.editLabelInput.focus()}},{key:"isUniqueLabel",value:function(t){return[].concat(i(this.getActiveTable().querySelectorAll("label"))).reduce((function(e,n){return e&&n.innerHTML!==t}),!0)}},{key:"saveLabel",value:function(){var t=arguments.length>0&&void 0!==arguments[0]&&arguments[0];if("none"!==this.editLabelInput.style.display){var e=this.editLabelInput.value.trim();if(""===e){var n=document.getElementById(this.EDIT_LABEL_ID).parentElement;return this.hideEditLabelInput(),n.parentNode.removeChild(n),void this.setSubmitStatus()}var r=this.editLabelInput.parentNode.querySelector("label");if(t||e===r.innerHTML)this.hideEditLabelInput();else if(this.isUniqueLabel(e)){var i=this.editLabelInput.parentNode.querySelector("input");this.hideEditLabelInput(),r.innerHTML=e,this.replaceName(i,e),this.setSubmitStatus()}else this.editLabelInput.classList.add(this.EDIT_LABEL_ERROR_CLASS)}}},{key:"replaceName",value:function(t,e){t.name=t.name.replace(/(.+\[.+])\[.*]/g,"$1["+e+"]")}},{key:"clearMessage",value:function(t){t.innerHTML="",t.classList.remove("active")}},{key:"clearMessages",value:function(){this.clearMessage(this.successMessage),this.clearMessage(this.errorMessage),clearTimeout(this.msgTimer)}},{key:"showMessage",value:function(t,e){var n=this;t.innerHTML=e,t.classList.add("active"),this.msgTimer=setTimeout((function(){n.clearMessages()}),5e3)}}]),t}();e.default=o}]);
2
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["webpack:///webpack/bootstrap","webpack:///./node_modules/cross-fetch/dist/browser-ponyfill.js","webpack:///./js/settings/app.js","webpack:///./js/settings/settings.js"],"names":["installedModules","__webpack_require__","moduleId","exports","module","i","l","modules","call","m","c","d","name","getter","o","Object","defineProperty","enumerable","get","r","Symbol","toStringTag","value","t","mode","__esModule","ns","create","key","bind","n","object","property","prototype","hasOwnProperty","p","s","__self__","root","F","this","fetch","DOMException","self","support","Blob","e","viewClasses","isArrayBufferView","ArrayBuffer","isView","obj","indexOf","toString","normalizeName","String","test","TypeError","toLowerCase","normalizeValue","iteratorFor","items","iterator","next","shift","done","undefined","Headers","headers","map","forEach","append","Array","isArray","header","getOwnPropertyNames","consumed","body","bodyUsed","Promise","reject","fileReaderReady","reader","resolve","onload","result","onerror","error","readBlobAsArrayBuffer","blob","FileReader","promise","readAsArrayBuffer","bufferClone","buf","slice","view","Uint8Array","byteLength","set","buffer","Body","_initBody","_bodyInit","_bodyText","isPrototypeOf","_bodyBlob","FormData","_bodyFormData","URLSearchParams","DataView","_bodyArrayBuffer","type","rejected","Error","arrayBuffer","then","text","readAsText","chars","length","fromCharCode","join","readArrayBufferAsText","formData","decode","json","JSON","parse","oldValue","has","callback","thisArg","keys","push","values","entries","methods","Request","input","options","method","upcased","url","credentials","signal","toUpperCase","referrer","form","trim","split","bytes","replace","decodeURIComponent","Response","bodyInit","status","ok","statusText","clone","response","redirectStatuses","redirect","RangeError","location","err","message","stack","constructor","init","request","aborted","xhr","XMLHttpRequest","abortXhr","abort","rawHeaders","getAllResponseHeaders","line","parts","responseURL","responseText","ontimeout","onabort","open","withCredentials","responseType","setRequestHeader","addEventListener","onreadystatechange","readyState","removeEventListener","send","polyfill","default","document","Settings","OPTIONS_FORM_SELECTOR","HEADER_SELECTOR","TABLE_SELECTOR","SUBMIT_SELECTOR","CONVERT_FORM_SELECTOR","CONVERT_BUTTON_SELECTOR","CONFIRM_POPUP_SELECTOR","CONFIRM_OK_SELECTOR","CONFIRM_CANCEL_SELECTOR","CURRENT_STUB_ID","CURRENT_NAV_TAB_CLASS","ACTIVE_NAV_TAB_CLASS","ACTIVE_TABLE_CLASS","EDIT_LABEL_ID","EDIT_LABEL_ERROR_CLASS","plusButton","PLUS_CLASS","optionsForm","querySelector","tablesData","getTablesData","submitButton","confirmPopup","optionsSaveSuccessMessage","Cyr2LatSettingsObject","optionsSaveErrorMessage","addWrapper","addMessageLines","addEditLabelInput","hideTables","bindEvents","setSubmitStatus","querySelectorAll","getActiveHeader","dataset","index","activeIndex","getActiveIndex","stringify","getActiveTableData","disabled","isActiveTableChanged","activeTable","getActiveTable","activeForm","createElement","action","getAttribute","appendChild","cloneNode","activeInputs","id","hiddenInputs","showMessage","successMessage","errorMessage","finally","parentNode","removeChild","table","inputs","data","label","innerHTML","getTables","getTableData","wrapper","classList","add","insertBefore","firstChild","addMessageLine","editLabelInput","style","display","remove","previousElementSibling","lastCell","getLastCell","parentElement","nextElementSibling","idArr","newId","parseInt","htmlFor","setAttribute","replaceName","editLabel","currentIndex","plus","getHeaders","onclick","event","preventDefault","target","saveActiveTable","aHeader","tables","getInputs","oninput","getLabels","onblur","saveLabel","onkeyup","getPlusButtons","addCell","hideConfirmPopup","stopPropagation","submit","focus","newValue","reduce","acc","cancel","editedCell","getElementById","hideEditLabelInput","isUniqueLabel","clearMessage","clearTimeout","msgTimer","el","setTimeout","clearMessages"],"mappings":"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","file":"js/settings/app.js","sourcesContent":[" \t// The module cache\n \tvar installedModules = {};\n\n \t// The require function\n \tfunction __webpack_require__(moduleId) {\n\n \t\t// Check if module is in cache\n \t\tif(installedModules[moduleId]) {\n \t\t\treturn installedModules[moduleId].exports;\n \t\t}\n \t\t// Create a new module (and put it into the cache)\n \t\tvar module = installedModules[moduleId] = {\n \t\t\ti: moduleId,\n \t\t\tl: false,\n \t\t\texports: {}\n \t\t};\n\n \t\t// Execute the module function\n \t\tmodules[moduleId].call(module.exports, module, module.exports, __webpack_require__);\n\n \t\t// Flag the module as loaded\n \t\tmodule.l = true;\n\n \t\t// Return the exports of the module\n \t\treturn module.exports;\n \t}\n\n\n \t// expose the modules object (__webpack_modules__)\n \t__webpack_require__.m = modules;\n\n \t// expose the module cache\n \t__webpack_require__.c = installedModules;\n\n \t// define getter function for harmony exports\n \t__webpack_require__.d = function(exports, name, getter) {\n \t\tif(!__webpack_require__.o(exports, name)) {\n \t\t\tObject.defineProperty(exports, name, { enumerable: true, get: getter });\n \t\t}\n \t};\n\n \t// define __esModule on exports\n \t__webpack_require__.r = function(exports) {\n \t\tif(typeof Symbol !== 'undefined' && Symbol.toStringTag) {\n \t\t\tObject.defineProperty(exports, Symbol.toStringTag, { value: 'Module' });\n \t\t}\n \t\tObject.defineProperty(exports, '__esModule', { value: true });\n \t};\n\n \t// create a fake namespace object\n \t// mode & 1: value is a module id, require it\n \t// mode & 2: merge all properties of value into the ns\n \t// mode & 4: return value when already ns object\n \t// mode & 8|1: behave like require\n \t__webpack_require__.t = function(value, mode) {\n \t\tif(mode & 1) value = __webpack_require__(value);\n \t\tif(mode & 8) return value;\n \t\tif((mode & 4) && typeof value === 'object' && value && value.__esModule) return value;\n \t\tvar ns = Object.create(null);\n \t\t__webpack_require__.r(ns);\n \t\tObject.defineProperty(ns, 'default', { enumerable: true, value: value });\n \t\tif(mode & 2 && typeof value != 'string') for(var key in value) __webpack_require__.d(ns, key, function(key) { return value[key]; }.bind(null, key));\n \t\treturn ns;\n \t};\n\n \t// getDefaultExport function for compatibility with non-harmony modules\n \t__webpack_require__.n = function(module) {\n \t\tvar getter = module && module.__esModule ?\n \t\t\tfunction getDefault() { return module['default']; } :\n \t\t\tfunction getModuleExports() { return module; };\n \t\t__webpack_require__.d(getter, 'a', getter);\n \t\treturn getter;\n \t};\n\n \t// Object.prototype.hasOwnProperty.call\n \t__webpack_require__.o = function(object, property) { return Object.prototype.hasOwnProperty.call(object, property); };\n\n \t// __webpack_public_path__\n \t__webpack_require__.p = \"\";\n\n\n \t// Load entry module and return exports\n \treturn __webpack_require__(__webpack_require__.s = 0);\n","var __self__ = (function (root) {\nfunction F() {\nthis.fetch = false;\nthis.DOMException = root.DOMException\n}\nF.prototype = root;\nreturn new F();\n})(typeof self !== 'undefined' ? self : this);\n(function(self) {\n\nvar irrelevant = (function (exports) {\n\n  var support = {\n    searchParams: 'URLSearchParams' in self,\n    iterable: 'Symbol' in self && 'iterator' in Symbol,\n    blob:\n      'FileReader' in self &&\n      'Blob' in self &&\n      (function() {\n        try {\n          new Blob();\n          return true\n        } catch (e) {\n          return false\n        }\n      })(),\n    formData: 'FormData' in self,\n    arrayBuffer: 'ArrayBuffer' in self\n  };\n\n  function isDataView(obj) {\n    return obj && DataView.prototype.isPrototypeOf(obj)\n  }\n\n  if (support.arrayBuffer) {\n    var viewClasses = [\n      '[object Int8Array]',\n      '[object Uint8Array]',\n      '[object Uint8ClampedArray]',\n      '[object Int16Array]',\n      '[object Uint16Array]',\n      '[object Int32Array]',\n      '[object Uint32Array]',\n      '[object Float32Array]',\n      '[object Float64Array]'\n    ];\n\n    var isArrayBufferView =\n      ArrayBuffer.isView ||\n      function(obj) {\n        return obj && viewClasses.indexOf(Object.prototype.toString.call(obj)) > -1\n      };\n  }\n\n  function normalizeName(name) {\n    if (typeof name !== 'string') {\n      name = String(name);\n    }\n    if (/[^a-z0-9\\-#$%&'*+.^_`|~]/i.test(name)) {\n      throw new TypeError('Invalid character in header field name')\n    }\n    return name.toLowerCase()\n  }\n\n  function normalizeValue(value) {\n    if (typeof value !== 'string') {\n      value = String(value);\n    }\n    return value\n  }\n\n  // Build a destructive iterator for the value list\n  function iteratorFor(items) {\n    var iterator = {\n      next: function() {\n        var value = items.shift();\n        return {done: value === undefined, value: value}\n      }\n    };\n\n    if (support.iterable) {\n      iterator[Symbol.iterator] = function() {\n        return iterator\n      };\n    }\n\n    return iterator\n  }\n\n  function Headers(headers) {\n    this.map = {};\n\n    if (headers instanceof Headers) {\n      headers.forEach(function(value, name) {\n        this.append(name, value);\n      }, this);\n    } else if (Array.isArray(headers)) {\n      headers.forEach(function(header) {\n        this.append(header[0], header[1]);\n      }, this);\n    } else if (headers) {\n      Object.getOwnPropertyNames(headers).forEach(function(name) {\n        this.append(name, headers[name]);\n      }, this);\n    }\n  }\n\n  Headers.prototype.append = function(name, value) {\n    name = normalizeName(name);\n    value = normalizeValue(value);\n    var oldValue = this.map[name];\n    this.map[name] = oldValue ? oldValue + ', ' + value : value;\n  };\n\n  Headers.prototype['delete'] = function(name) {\n    delete this.map[normalizeName(name)];\n  };\n\n  Headers.prototype.get = function(name) {\n    name = normalizeName(name);\n    return this.has(name) ? this.map[name] : null\n  };\n\n  Headers.prototype.has = function(name) {\n    return this.map.hasOwnProperty(normalizeName(name))\n  };\n\n  Headers.prototype.set = function(name, value) {\n    this.map[normalizeName(name)] = normalizeValue(value);\n  };\n\n  Headers.prototype.forEach = function(callback, thisArg) {\n    for (var name in this.map) {\n      if (this.map.hasOwnProperty(name)) {\n        callback.call(thisArg, this.map[name], name, this);\n      }\n    }\n  };\n\n  Headers.prototype.keys = function() {\n    var items = [];\n    this.forEach(function(value, name) {\n      items.push(name);\n    });\n    return iteratorFor(items)\n  };\n\n  Headers.prototype.values = function() {\n    var items = [];\n    this.forEach(function(value) {\n      items.push(value);\n    });\n    return iteratorFor(items)\n  };\n\n  Headers.prototype.entries = function() {\n    var items = [];\n    this.forEach(function(value, name) {\n      items.push([name, value]);\n    });\n    return iteratorFor(items)\n  };\n\n  if (support.iterable) {\n    Headers.prototype[Symbol.iterator] = Headers.prototype.entries;\n  }\n\n  function consumed(body) {\n    if (body.bodyUsed) {\n      return Promise.reject(new TypeError('Already read'))\n    }\n    body.bodyUsed = true;\n  }\n\n  function fileReaderReady(reader) {\n    return new Promise(function(resolve, reject) {\n      reader.onload = function() {\n        resolve(reader.result);\n      };\n      reader.onerror = function() {\n        reject(reader.error);\n      };\n    })\n  }\n\n  function readBlobAsArrayBuffer(blob) {\n    var reader = new FileReader();\n    var promise = fileReaderReady(reader);\n    reader.readAsArrayBuffer(blob);\n    return promise\n  }\n\n  function readBlobAsText(blob) {\n    var reader = new FileReader();\n    var promise = fileReaderReady(reader);\n    reader.readAsText(blob);\n    return promise\n  }\n\n  function readArrayBufferAsText(buf) {\n    var view = new Uint8Array(buf);\n    var chars = new Array(view.length);\n\n    for (var i = 0; i < view.length; i++) {\n      chars[i] = String.fromCharCode(view[i]);\n    }\n    return chars.join('')\n  }\n\n  function bufferClone(buf) {\n    if (buf.slice) {\n      return buf.slice(0)\n    } else {\n      var view = new Uint8Array(buf.byteLength);\n      view.set(new Uint8Array(buf));\n      return view.buffer\n    }\n  }\n\n  function Body() {\n    this.bodyUsed = false;\n\n    this._initBody = function(body) {\n      this._bodyInit = body;\n      if (!body) {\n        this._bodyText = '';\n      } else if (typeof body === 'string') {\n        this._bodyText = body;\n      } else if (support.blob && Blob.prototype.isPrototypeOf(body)) {\n        this._bodyBlob = body;\n      } else if (support.formData && FormData.prototype.isPrototypeOf(body)) {\n        this._bodyFormData = body;\n      } else if (support.searchParams && URLSearchParams.prototype.isPrototypeOf(body)) {\n        this._bodyText = body.toString();\n      } else if (support.arrayBuffer && support.blob && isDataView(body)) {\n        this._bodyArrayBuffer = bufferClone(body.buffer);\n        // IE 10-11 can't handle a DataView body.\n        this._bodyInit = new Blob([this._bodyArrayBuffer]);\n      } else if (support.arrayBuffer && (ArrayBuffer.prototype.isPrototypeOf(body) || isArrayBufferView(body))) {\n        this._bodyArrayBuffer = bufferClone(body);\n      } else {\n        this._bodyText = body = Object.prototype.toString.call(body);\n      }\n\n      if (!this.headers.get('content-type')) {\n        if (typeof body === 'string') {\n          this.headers.set('content-type', 'text/plain;charset=UTF-8');\n        } else if (this._bodyBlob && this._bodyBlob.type) {\n          this.headers.set('content-type', this._bodyBlob.type);\n        } else if (support.searchParams && URLSearchParams.prototype.isPrototypeOf(body)) {\n          this.headers.set('content-type', 'application/x-www-form-urlencoded;charset=UTF-8');\n        }\n      }\n    };\n\n    if (support.blob) {\n      this.blob = function() {\n        var rejected = consumed(this);\n        if (rejected) {\n          return rejected\n        }\n\n        if (this._bodyBlob) {\n          return Promise.resolve(this._bodyBlob)\n        } else if (this._bodyArrayBuffer) {\n          return Promise.resolve(new Blob([this._bodyArrayBuffer]))\n        } else if (this._bodyFormData) {\n          throw new Error('could not read FormData body as blob')\n        } else {\n          return Promise.resolve(new Blob([this._bodyText]))\n        }\n      };\n\n      this.arrayBuffer = function() {\n        if (this._bodyArrayBuffer) {\n          return consumed(this) || Promise.resolve(this._bodyArrayBuffer)\n        } else {\n          return this.blob().then(readBlobAsArrayBuffer)\n        }\n      };\n    }\n\n    this.text = function() {\n      var rejected = consumed(this);\n      if (rejected) {\n        return rejected\n      }\n\n      if (this._bodyBlob) {\n        return readBlobAsText(this._bodyBlob)\n      } else if (this._bodyArrayBuffer) {\n        return Promise.resolve(readArrayBufferAsText(this._bodyArrayBuffer))\n      } else if (this._bodyFormData) {\n        throw new Error('could not read FormData body as text')\n      } else {\n        return Promise.resolve(this._bodyText)\n      }\n    };\n\n    if (support.formData) {\n      this.formData = function() {\n        return this.text().then(decode)\n      };\n    }\n\n    this.json = function() {\n      return this.text().then(JSON.parse)\n    };\n\n    return this\n  }\n\n  // HTTP methods whose capitalization should be normalized\n  var methods = ['DELETE', 'GET', 'HEAD', 'OPTIONS', 'POST', 'PUT'];\n\n  function normalizeMethod(method) {\n    var upcased = method.toUpperCase();\n    return methods.indexOf(upcased) > -1 ? upcased : method\n  }\n\n  function Request(input, options) {\n    options = options || {};\n    var body = options.body;\n\n    if (input instanceof Request) {\n      if (input.bodyUsed) {\n        throw new TypeError('Already read')\n      }\n      this.url = input.url;\n      this.credentials = input.credentials;\n      if (!options.headers) {\n        this.headers = new Headers(input.headers);\n      }\n      this.method = input.method;\n      this.mode = input.mode;\n      this.signal = input.signal;\n      if (!body && input._bodyInit != null) {\n        body = input._bodyInit;\n        input.bodyUsed = true;\n      }\n    } else {\n      this.url = String(input);\n    }\n\n    this.credentials = options.credentials || this.credentials || 'same-origin';\n    if (options.headers || !this.headers) {\n      this.headers = new Headers(options.headers);\n    }\n    this.method = normalizeMethod(options.method || this.method || 'GET');\n    this.mode = options.mode || this.mode || null;\n    this.signal = options.signal || this.signal;\n    this.referrer = null;\n\n    if ((this.method === 'GET' || this.method === 'HEAD') && body) {\n      throw new TypeError('Body not allowed for GET or HEAD requests')\n    }\n    this._initBody(body);\n  }\n\n  Request.prototype.clone = function() {\n    return new Request(this, {body: this._bodyInit})\n  };\n\n  function decode(body) {\n    var form = new FormData();\n    body\n      .trim()\n      .split('&')\n      .forEach(function(bytes) {\n        if (bytes) {\n          var split = bytes.split('=');\n          var name = split.shift().replace(/\\+/g, ' ');\n          var value = split.join('=').replace(/\\+/g, ' ');\n          form.append(decodeURIComponent(name), decodeURIComponent(value));\n        }\n      });\n    return form\n  }\n\n  function parseHeaders(rawHeaders) {\n    var headers = new Headers();\n    // Replace instances of \\r\\n and \\n followed by at least one space or horizontal tab with a space\n    // https://tools.ietf.org/html/rfc7230#section-3.2\n    var preProcessedHeaders = rawHeaders.replace(/\\r?\\n[\\t ]+/g, ' ');\n    preProcessedHeaders.split(/\\r?\\n/).forEach(function(line) {\n      var parts = line.split(':');\n      var key = parts.shift().trim();\n      if (key) {\n        var value = parts.join(':').trim();\n        headers.append(key, value);\n      }\n    });\n    return headers\n  }\n\n  Body.call(Request.prototype);\n\n  function Response(bodyInit, options) {\n    if (!options) {\n      options = {};\n    }\n\n    this.type = 'default';\n    this.status = options.status === undefined ? 200 : options.status;\n    this.ok = this.status >= 200 && this.status < 300;\n    this.statusText = 'statusText' in options ? options.statusText : 'OK';\n    this.headers = new Headers(options.headers);\n    this.url = options.url || '';\n    this._initBody(bodyInit);\n  }\n\n  Body.call(Response.prototype);\n\n  Response.prototype.clone = function() {\n    return new Response(this._bodyInit, {\n      status: this.status,\n      statusText: this.statusText,\n      headers: new Headers(this.headers),\n      url: this.url\n    })\n  };\n\n  Response.error = function() {\n    var response = new Response(null, {status: 0, statusText: ''});\n    response.type = 'error';\n    return response\n  };\n\n  var redirectStatuses = [301, 302, 303, 307, 308];\n\n  Response.redirect = function(url, status) {\n    if (redirectStatuses.indexOf(status) === -1) {\n      throw new RangeError('Invalid status code')\n    }\n\n    return new Response(null, {status: status, headers: {location: url}})\n  };\n\n  exports.DOMException = self.DOMException;\n  try {\n    new exports.DOMException();\n  } catch (err) {\n    exports.DOMException = function(message, name) {\n      this.message = message;\n      this.name = name;\n      var error = Error(message);\n      this.stack = error.stack;\n    };\n    exports.DOMException.prototype = Object.create(Error.prototype);\n    exports.DOMException.prototype.constructor = exports.DOMException;\n  }\n\n  function fetch(input, init) {\n    return new Promise(function(resolve, reject) {\n      var request = new Request(input, init);\n\n      if (request.signal && request.signal.aborted) {\n        return reject(new exports.DOMException('Aborted', 'AbortError'))\n      }\n\n      var xhr = new XMLHttpRequest();\n\n      function abortXhr() {\n        xhr.abort();\n      }\n\n      xhr.onload = function() {\n        var options = {\n          status: xhr.status,\n          statusText: xhr.statusText,\n          headers: parseHeaders(xhr.getAllResponseHeaders() || '')\n        };\n        options.url = 'responseURL' in xhr ? xhr.responseURL : options.headers.get('X-Request-URL');\n        var body = 'response' in xhr ? xhr.response : xhr.responseText;\n        resolve(new Response(body, options));\n      };\n\n      xhr.onerror = function() {\n        reject(new TypeError('Network request failed'));\n      };\n\n      xhr.ontimeout = function() {\n        reject(new TypeError('Network request failed'));\n      };\n\n      xhr.onabort = function() {\n        reject(new exports.DOMException('Aborted', 'AbortError'));\n      };\n\n      xhr.open(request.method, request.url, true);\n\n      if (request.credentials === 'include') {\n        xhr.withCredentials = true;\n      } else if (request.credentials === 'omit') {\n        xhr.withCredentials = false;\n      }\n\n      if ('responseType' in xhr && support.blob) {\n        xhr.responseType = 'blob';\n      }\n\n      request.headers.forEach(function(value, name) {\n        xhr.setRequestHeader(name, value);\n      });\n\n      if (request.signal) {\n        request.signal.addEventListener('abort', abortXhr);\n\n        xhr.onreadystatechange = function() {\n          // DONE (success or failure)\n          if (xhr.readyState === 4) {\n            request.signal.removeEventListener('abort', abortXhr);\n          }\n        };\n      }\n\n      xhr.send(typeof request._bodyInit === 'undefined' ? null : request._bodyInit);\n    })\n  }\n\n  fetch.polyfill = true;\n\n  if (!self.fetch) {\n    self.fetch = fetch;\n    self.Headers = Headers;\n    self.Request = Request;\n    self.Response = Response;\n  }\n\n  exports.Headers = Headers;\n  exports.Request = Request;\n  exports.Response = Response;\n  exports.fetch = fetch;\n\n  return exports;\n\n}({}));\n})(__self__);\ndelete __self__.fetch.polyfill\nexports = __self__.fetch // To enable: import fetch from 'cross-fetch'\nexports.default = __self__.fetch // For TypeScript consumers without esModuleInterop.\nexports.fetch = __self__.fetch // To enable: import {fetch} from 'cross-fetch'\nexports.Headers = __self__.Headers\nexports.Request = __self__.Request\nexports.Response = __self__.Response\nmodule.exports = exports\n","/**\n * @file Settings Application.\n */\nimport Settings from './settings';\n\ndocument.addEventListener( 'DOMContentLoaded', () => {\n\tnew Settings();\n} );\n","/**\n * @file class Settings.\n */\n\n/* global Cyr2LatSettingsObject */\n\nclass Settings {\n\t/**\n\t * Class constructor.\n\t */\n\tconstructor() {\n\t\tthis.OPTIONS_FORM_SELECTOR = '#ctl-options';\n\t\tthis.HEADER_SELECTOR = this.OPTIONS_FORM_SELECTOR + ' h2';\n\t\tthis.TABLE_SELECTOR = this.OPTIONS_FORM_SELECTOR + ' table';\n\t\tthis.SUBMIT_SELECTOR = this.OPTIONS_FORM_SELECTOR + ' #submit';\n\t\tthis.CONVERT_FORM_SELECTOR = '#ctl-convert-existing-slugs';\n\t\tthis.CONVERT_BUTTON_SELECTOR = '#ctl-convert-button';\n\t\tthis.CONFIRM_POPUP_SELECTOR = '#ctl-confirm-popup';\n\t\tthis.CONFIRM_OK_SELECTOR = '#ctl-confirm-ok';\n\t\tthis.CONFIRM_CANCEL_SELECTOR = '#ctl-confirm-cancel';\n\t\tthis.CURRENT_STUB_ID = 'ctl-current';\n\t\tthis.CURRENT_NAV_TAB_CLASS = 'nav-tab-current';\n\t\tthis.ACTIVE_NAV_TAB_CLASS = 'nav-tab-active';\n\t\tthis.ACTIVE_TABLE_CLASS = 'active';\n\t\tthis.EDIT_LABEL_ID = 'ctl-edit-label';\n\t\tthis.EDIT_LABEL_ERROR_CLASS = 'ctl-edit-label-error';\n\t\tthis.plusButton =\n\t\t\t'<button type=\"button\" aria-haspopup=\"true\" aria-expanded=\"false\" class=\"components-button block-editor-inserter__toggle has-icon\" aria-label=\"Добавить блок\">' +\n\t\t\t'<svg width=\"24\" height=\"24\" xmlns=\"http://www.w3.org/2000/svg\" viewBox=\"-2 -2 24 24\" role=\"img\" aria-hidden=\"true\" focusable=\"false\">' +\n\t\t\t'<path d=\"M10 1c-5 0-9 4-9 9s4 9 9 9 9-4 9-9-4-9-9-9zm0 16c-3.9 0-7-3.1-7-7s3.1-7 7-7 7 3.1 7 7-3.1 7-7 7zm1-11H9v3H6v2h3v3h2v-3h3V9h-3V6zM10 1c-5 0-9 4-9 9s4 9 9 9 9-4 9-9-4-9-9-9zm0 16c-3.9 0-7-3.1-7-7s3.1-7 7-7 7 3.1 7 7-3.1 7-7 7zm1-11H9v3H6v2h3v3h2v-3h3V9h-3V6z\">' +\n\t\t\t'</path>' +\n\t\t\t'</svg>' +\n\t\t\t'</button>';\n\t\tthis.PLUS_CLASS = 'ctl-plus';\n\n\t\tthis.optionsForm = document.querySelector( this.OPTIONS_FORM_SELECTOR );\n\t\tthis.tablesData = this.getTablesData();\n\t\tthis.submitButton = document.querySelector( this.SUBMIT_SELECTOR );\n\t\tthis.confirmPopup = document.querySelector(\n\t\t\tthis.CONFIRM_POPUP_SELECTOR\n\t\t);\n\n\t\t// Copy to class properties, otherwise eslint marks some properties of global object as unresolved.\n\t\tthis.optionsSaveSuccessMessage =\n\t\t\tCyr2LatSettingsObject.optionsSaveSuccessMessage;\n\t\tthis.optionsSaveErrorMessage =\n\t\t\tCyr2LatSettingsObject.optionsSaveErrorMessage;\n\n\t\tthis.addWrapper();\n\t\tthis.addMessageLines();\n\t\tthis.addEditLabelInput();\n\t\tthis.hideTables();\n\t\tthis.bindEvents();\n\t\tthis.setSubmitStatus();\n\t}\n\n\t/**\n\t * Get headers.\n\t *\n\t * @return {*[]} Headers.\n\t */\n\tgetHeaders() {\n\t\treturn [ ...document.querySelectorAll( this.HEADER_SELECTOR ) ];\n\t}\n\n\t/**\n\t * Get active header.\n\t *\n\t * @return {Element} Active header.\n\t */\n\tgetActiveHeader() {\n\t\treturn document.querySelector(\n\t\t\tthis.HEADER_SELECTOR + '.' + this.ACTIVE_NAV_TAB_CLASS\n\t\t);\n\t}\n\n\t/**\n\t * Get active index.\n\t *\n\t * @return {*} Active index.\n\t */\n\tgetActiveIndex() {\n\t\treturn this.getActiveHeader().dataset.index;\n\t}\n\n\t/**\n\t * Get tables.\n\t *\n\t * @return {*[]} Tables.\n\t */\n\tgetTables() {\n\t\treturn [ ...document.querySelectorAll( this.TABLE_SELECTOR ) ];\n\t}\n\n\t/**\n\t * Get active table.\n\t *\n\t * @return {Element} Active table.\n\t */\n\tgetActiveTable() {\n\t\treturn document.querySelector(\n\t\t\tthis.TABLE_SELECTOR + '.' + this.ACTIVE_TABLE_CLASS\n\t\t);\n\t}\n\n\t/**\n\t * Get inputs.\n\t *\n\t * @return {*[]} Inputs.\n\t */\n\tgetInputs() {\n\t\treturn [\n\t\t\t...document.querySelectorAll(\n\t\t\t\tthis.OPTIONS_FORM_SELECTOR + ' input'\n\t\t\t),\n\t\t];\n\t}\n\n\t/**\n\t * Get labels.\n\t *\n\t * @return {*[]} Labels.\n\t */\n\tgetLabels() {\n\t\treturn [\n\t\t\t...document.querySelectorAll(\n\t\t\t\tthis.OPTIONS_FORM_SELECTOR + ' label'\n\t\t\t),\n\t\t];\n\t}\n\n\t/**\n\t * Get plus buttons.\n\t *\n\t * @return {*[]} Plus buttons.\n\t */\n\tgetPlusButtons() {\n\t\treturn [\n\t\t\t...document.querySelectorAll(\n\t\t\t\tthis.OPTIONS_FORM_SELECTOR + ' .' + this.PLUS_CLASS\n\t\t\t),\n\t\t];\n\t}\n\n\t/**\n\t * Check of active table was changed.\n\t *\n\t * @return {boolean} If active table was changed.\n\t */\n\tisActiveTableChanged() {\n\t\tconst activeIndex = this.getActiveIndex();\n\n\t\treturn (\n\t\t\tJSON.stringify( this.getActiveTableData() ) !==\n\t\t\tJSON.stringify( this.tablesData[ activeIndex ] )\n\t\t);\n\t}\n\n\t/**\n\t * Set status of submit button.\n\t */\n\tsetSubmitStatus() {\n\t\tthis.submitButton.disabled = ! this.isActiveTableChanged();\n\t}\n\n\t/**\n\t * Save active table.\n\t */\n\tsaveActiveTable() {\n\t\tif ( ! this.isActiveTableChanged() ) {\n\t\t\treturn;\n\t\t}\n\n\t\tconst activeTable = this.getActiveTable();\n\n\t\tconst activeForm = document.createElement( 'form' );\n\t\tactiveForm.action = this.optionsForm.getAttribute( 'action' );\n\t\tactiveForm.method = this.optionsForm.method;\n\t\tactiveForm.appendChild( activeTable.cloneNode( true ) );\n\n\t\tconst activeInputs = [ ...activeTable.querySelectorAll( 'input' ) ];\n\t\tactiveInputs.map( ( input ) => {\n\t\t\treturn ( activeForm.querySelector( '#' + input.id ).value =\n\t\t\t\tinput.value );\n\t\t} );\n\n\t\tconst hiddenInputs = [\n\t\t\t...this.optionsForm.querySelectorAll( 'input[type=\"hidden\"]' ),\n\t\t];\n\t\thiddenInputs.map( ( input ) => {\n\t\t\treturn activeForm.appendChild( input.cloneNode( true ) );\n\t\t} );\n\t\tdocument.body.appendChild( activeForm );\n\n\t\treturn fetch( this.optionsForm.getAttribute( 'action' ), {\n\t\t\tmethod: activeForm.method,\n\t\t\tbody: new URLSearchParams( [ ...new FormData( activeForm ) ] ),\n\t\t} )\n\t\t\t.then( ( response ) => {\n\t\t\t\tif ( response.ok ) {\n\t\t\t\t\tthis.showMessage(\n\t\t\t\t\t\tthis.successMessage,\n\t\t\t\t\t\tthis.optionsSaveSuccessMessage\n\t\t\t\t\t);\n\t\t\t\t\tthis.tablesData = this.getTablesData();\n\t\t\t\t} else {\n\t\t\t\t\tthis.showMessage(\n\t\t\t\t\t\tthis.errorMessage,\n\t\t\t\t\t\tthis.optionsSaveErrorMessage\n\t\t\t\t\t);\n\t\t\t\t}\n\n\t\t\t\treturn response.json();\n\t\t\t} )\n\t\t\t.finally( () => {\n\t\t\t\tactiveForm.parentNode.removeChild( activeForm );\n\t\t\t\tthis.setSubmitStatus();\n\t\t\t} );\n\t}\n\n\t/**\n\t * Get table data.\n\t *\n\t * @param {Element} table Table.\n\t * @return {{}[]} Table data.\n\t */\n\tgetTableData( table ) {\n\t\tconst inputs = [ ...table.querySelectorAll( 'input' ) ];\n\n\t\tconst data = {};\n\t\tinputs.forEach( ( input ) => {\n\t\t\tconst label = document.querySelector(\n\t\t\t\tthis.OPTIONS_FORM_SELECTOR + ' label[for=\"' + input.id + '\"]'\n\t\t\t);\n\n\t\t\tdata[ label.innerHTML ] = input.value;\n\t\t} );\n\n\t\treturn data;\n\t}\n\n\t/**\n\t * Get data from all tables.\n\t *\n\t * @return {{}[][]} Data from all tables.\n\t */\n\tgetTablesData() {\n\t\treturn this.getTables().map( ( table ) => {\n\t\t\treturn this.getTableData( table );\n\t\t} );\n\t}\n\n\t/**\n\t * Get active table data.\n\t *\n\t * @return {{}[]} Active table data.\n\t */\n\tgetActiveTableData() {\n\t\treturn this.getTableData( this.getActiveTable() );\n\t}\n\n\t/**\n\t * Add wrapper.\n\t */\n\taddWrapper() {\n\t\tthis.wrapper = document.createElement( 'ul' );\n\t\tthis.wrapper.classList.add( 'nav-tab-wrapper' );\n\t\tthis.optionsForm.insertBefore(\n\t\t\tthis.wrapper,\n\t\t\tthis.optionsForm.firstChild\n\t\t);\n\t}\n\n\t/**\n\t * Add message line.\n\t *\n\t * @param {string} id Message id.\n\t * @return {HTMLDivElement} Message line.\n\t */\n\taddMessageLine( id ) {\n\t\tconst message = document.createElement( 'div' );\n\t\tmessage.id = id;\n\t\tthis.optionsForm.insertBefore( message, this.optionsForm.firstChild );\n\n\t\treturn message;\n\t}\n\n\t/**\n\t * Add success and error message lines.\n\t */\n\taddMessageLines() {\n\t\tthis.successMessage = this.addMessageLine( 'ctl-success' );\n\t\tthis.errorMessage = this.addMessageLine( 'ctl-error' );\n\t}\n\n\t/**\n\t * Add edit label input.\n\t */\n\taddEditLabelInput() {\n\t\tthis.editLabelInput = document.createElement( 'input' );\n\t\tthis.editLabelInput.id = this.EDIT_LABEL_ID;\n\t\tthis.editLabelInput.style.display = 'none';\n\t\tdocument.body.appendChild( this.editLabelInput );\n\t}\n\n\t/**\n\t * Hide edit label input.\n\t */\n\thideEditLabelInput() {\n\t\tthis.editLabelInput.style.display = 'none';\n\t\tthis.editLabelInput.classList.remove( this.EDIT_LABEL_ERROR_CLASS );\n\t\tdocument.body.appendChild( this.editLabelInput );\n\t}\n\n\t/**\n\t * Get last cell in active table.\n\t *\n\t * @return {Element} Last cell.\n\t */\n\tgetLastCell() {\n\t\treturn document.querySelector(\n\t\t\tthis.OPTIONS_FORM_SELECTOR +\n\t\t\t\t' .' +\n\t\t\t\tthis.ACTIVE_TABLE_CLASS +\n\t\t\t\t' .' +\n\t\t\t\tthis.PLUS_CLASS\n\t\t).previousElementSibling;\n\t}\n\n\t/**\n\t * Add new cell to the active table.\n\t */\n\taddCell() {\n\t\tlet lastCell = this.getLastCell();\n\t\tlastCell.parentElement.insertBefore(\n\t\t\tlastCell.cloneNode( true ),\n\t\t\tlastCell.nextElementSibling\n\t\t);\n\n\t\tlastCell = this.getLastCell();\n\t\tconst label = lastCell.querySelector( 'label' );\n\t\tconst input = lastCell.querySelector( 'input' );\n\n\t\tconst idArr = input.id.split( '-' );\n\t\tconst newId = idArr[ 0 ] + '-' + ( parseInt( idArr[ 1 ] ) + 1 );\n\n\t\tlabel.htmlFor = newId;\n\t\tlabel.innerHTML = '';\n\n\t\tinput.id = newId;\n\t\tinput.value = '';\n\t\tinput.setAttribute( 'value', '' );\n\t\tthis.replaceName( input, '' );\n\n\t\tthis.bindEvents();\n\n\t\tthis.editLabel( label );\n\t}\n\n\t/**\n\t * Hide conversion tables except the first one.\n\t * Create navigation tabs.\n\t */\n\thideTables() {\n\t\tlet currentIndex = 0;\n\n\t\tthis.getTables().map( ( table, index ) => {\n\t\t\ttable.classList.add( 'ctl-table' );\n\n\t\t\tif ( this.CURRENT_STUB_ID === table.previousElementSibling.id ) {\n\t\t\t\tcurrentIndex = index;\n\t\t\t\ttable.classList.add( this.ACTIVE_TABLE_CLASS );\n\t\t\t}\n\n\t\t\tconst plus = document.createElement( 'div' );\n\t\t\tplus.classList.add( this.PLUS_CLASS );\n\t\t\tplus.innerHTML = this.plusButton;\n\t\t\ttable.querySelector( 'td' ).appendChild( plus );\n\n\t\t\treturn null;\n\t\t} );\n\n\t\tthis.getHeaders().map( ( header, index ) => {\n\t\t\theader.classList.add( 'nav-tab' );\n\t\t\theader.dataset.index = index;\n\n\t\t\tthis.wrapper.appendChild( header );\n\t\t\tif ( index === currentIndex ) {\n\t\t\t\theader.classList.add( this.CURRENT_NAV_TAB_CLASS );\n\t\t\t\theader.classList.add( this.ACTIVE_NAV_TAB_CLASS );\n\t\t\t}\n\n\t\t\treturn null;\n\t\t} );\n\t}\n\n\t/**\n\t * Bind events to methods.\n\t */\n\tbindEvents() {\n\t\tthis.getHeaders().map( ( header, i, headers ) => {\n\t\t\theader.onclick = ( event ) => {\n\t\t\t\tevent.preventDefault();\n\n\t\t\t\tconst index = event.target.dataset.index;\n\t\t\t\tconst activeIndex = this.getActiveIndex();\n\n\t\t\t\tif ( index === activeIndex ) {\n\t\t\t\t\treturn false;\n\t\t\t\t}\n\n\t\t\t\tthis.saveActiveTable();\n\n\t\t\t\theaders.map( ( aHeader ) => {\n\t\t\t\t\treturn aHeader.classList.remove(\n\t\t\t\t\t\tthis.ACTIVE_NAV_TAB_CLASS\n\t\t\t\t\t);\n\t\t\t\t} );\n\t\t\t\theaders[ index ].classList.add( this.ACTIVE_NAV_TAB_CLASS );\n\n\t\t\t\tconst tables = this.getTables();\n\t\t\t\ttables.map( ( table ) => {\n\t\t\t\t\treturn table.classList.remove( this.ACTIVE_TABLE_CLASS );\n\t\t\t\t} );\n\t\t\t\ttables[ index ].classList.add( this.ACTIVE_TABLE_CLASS );\n\n\t\t\t\tthis.setSubmitStatus();\n\n\t\t\t\treturn false;\n\t\t\t};\n\n\t\t\treturn null;\n\t\t} );\n\n\t\tthis.getInputs().map( ( input ) => {\n\t\t\tinput.oninput = () => {\n\t\t\t\tthis.setSubmitStatus();\n\t\t\t};\n\n\t\t\treturn null;\n\t\t} );\n\n\t\tthis.getLabels().map( ( label ) => {\n\t\t\tlabel.onclick = ( event ) => {\n\t\t\t\tevent.preventDefault();\n\t\t\t\tthis.editLabel( event.target );\n\t\t\t\treturn false;\n\t\t\t};\n\n\t\t\treturn null;\n\t\t} );\n\n\t\tthis.editLabelInput.onblur = () => {\n\t\t\tthis.saveLabel();\n\t\t};\n\n\t\tthis.editLabelInput.onkeyup = ( e ) => {\n\t\t\tif ( 'Escape' === e.key ) {\n\t\t\t\tthis.saveLabel( true );\n\t\t\t}\n\n\t\t\tif ( 'Enter' === e.key ) {\n\t\t\t\tthis.saveLabel();\n\t\t\t}\n\t\t};\n\n\t\tthis.getPlusButtons().map( ( plus ) => {\n\t\t\tplus.onclick = ( event ) => {\n\t\t\t\tevent.preventDefault();\n\t\t\t\tthis.addCell( event.target );\n\t\t\t\treturn false;\n\t\t\t};\n\n\t\t\treturn null;\n\t\t} );\n\n\t\tthis.submitButton.onclick = ( event ) => {\n\t\t\tevent.preventDefault();\n\t\t\tthis.saveActiveTable();\n\t\t\treturn false;\n\t\t};\n\n\t\tdocument.querySelector( this.CONVERT_BUTTON_SELECTOR ).onclick = (\n\t\t\tevent\n\t\t) => {\n\t\t\tevent.preventDefault();\n\t\t\tthis.confirmPopup.style.display = 'block';\n\t\t\treturn false;\n\t\t};\n\n\t\tthis.confirmPopup.onclick = () => {\n\t\t\tthis.hideConfirmPopup();\n\t\t};\n\n\t\tdocument.querySelector( this.CONFIRM_OK_SELECTOR ).onclick = (\n\t\t\tevent\n\t\t) => {\n\t\t\tevent.stopPropagation();\n\t\t\tthis.hideConfirmPopup();\n\t\t\tdocument.querySelector( this.CONVERT_FORM_SELECTOR ).submit();\n\t\t};\n\n\t\tdocument.querySelector( this.CONFIRM_CANCEL_SELECTOR ).onclick = (\n\t\t\tevent\n\t\t) => {\n\t\t\tevent.stopPropagation();\n\t\t\tthis.hideConfirmPopup();\n\t\t};\n\t}\n\n\t/**\n\t * Hide confirmation popup.\n\t */\n\thideConfirmPopup() {\n\t\tthis.confirmPopup.style.display = 'none';\n\t}\n\n\t/**\n\t * Edit label.\n\t *\n\t * @param {Element} label Label to edit.\n\t */\n\teditLabel( label ) {\n\t\tlabel.parentNode.appendChild( this.editLabelInput );\n\t\tthis.editLabelInput.value = label.innerHTML;\n\n\t\tthis.editLabelInput.classList.remove( this.EDIT_LABEL_ERROR_CLASS );\n\t\tthis.editLabelInput.style.display = 'block';\n\t\tthis.editLabelInput.focus();\n\t}\n\n\t/**\n\t * Is new value of edited label unique in active table.\n\t *\n\t * @param {string} newValue New Value from edited label.\n\t * @return {*} If new value of edited label is unique in active table.\n\t */\n\tisUniqueLabel( newValue ) {\n\t\treturn [ ...this.getActiveTable().querySelectorAll( 'label' ) ].reduce(\n\t\t\t( acc, label ) => {\n\t\t\t\treturn acc && label.innerHTML !== newValue;\n\t\t\t},\n\t\t\ttrue\n\t\t);\n\t}\n\n\t/**\n\t * Save modified label.\n\t *\n\t * @param {boolean} cancel Cancel.\n\t */\n\tsaveLabel( cancel = false ) {\n\t\tif ( 'none' === this.editLabelInput.style.display ) {\n\t\t\treturn;\n\t\t}\n\n\t\tconst newValue = this.editLabelInput.value.trim();\n\n\t\tif ( '' === newValue ) {\n\t\t\tconst editedCell = document.getElementById( this.EDIT_LABEL_ID )\n\t\t\t\t.parentElement;\n\t\t\tthis.hideEditLabelInput();\n\t\t\teditedCell.parentNode.removeChild( editedCell );\n\t\t\tthis.setSubmitStatus();\n\n\t\t\treturn;\n\t\t}\n\n\t\tconst label = this.editLabelInput.parentNode.querySelector( 'label' );\n\t\tif ( cancel || newValue === label.innerHTML ) {\n\t\t\tthis.hideEditLabelInput();\n\n\t\t\treturn;\n\t\t}\n\n\t\tif ( ! this.isUniqueLabel( newValue ) ) {\n\t\t\tthis.editLabelInput.classList.add( this.EDIT_LABEL_ERROR_CLASS );\n\n\t\t\treturn;\n\t\t}\n\n\t\tconst input = this.editLabelInput.parentNode.querySelector( 'input' );\n\n\t\tthis.hideEditLabelInput();\n\n\t\tlabel.innerHTML = newValue;\n\t\tthis.replaceName( input, newValue );\n\n\t\tthis.setSubmitStatus();\n\t}\n\n\t/**\n\t * Replace input name according to the new label value.\n\t *\n\t * @param {Element} input Input\n\t * @param {string} newValue New label value\n\t */\n\treplaceName( input, newValue ) {\n\t\t// noinspection JSUnresolvedVariable\n\t\tinput.name = input.name.replace(\n\t\t\t/(.+\\[.+])\\[.*]/g,\n\t\t\t'$1[' + newValue + ']'\n\t\t);\n\t}\n\n\t/**\n\t * Clear message.\n\t *\n\t * @param {HTMLDivElement} message Message.\n\t */\n\tclearMessage( message ) {\n\t\tmessage.innerHTML = '';\n\t\tmessage.classList.remove( 'active' );\n\t}\n\n\t/**\n\t * Clear messages.\n\t */\n\tclearMessages() {\n\t\tthis.clearMessage( this.successMessage );\n\t\tthis.clearMessage( this.errorMessage );\n\t\tclearTimeout( this.msgTimer );\n\t}\n\n\t/**\n\t * Show message.\n\t *\n\t * @param {HTMLDivElement} el Element.\n\t * @param {string} message Message.\n\t */\n\tshowMessage( el, message ) {\n\t\tel.innerHTML = message;\n\t\tel.classList.add( 'active' );\n\n\t\tthis.msgTimer = setTimeout( () => {\n\t\t\tthis.clearMessages();\n\t\t}, 5000 );\n\t}\n}\n\nexport default Settings;\n"],"sourceRoot":""}
 
 
languages/cyr2lat-ru_RU.mo CHANGED
Binary file
languages/cyr2lat-ru_RU.po CHANGED
@@ -1,8 +1,8 @@
1
  msgid ""
2
  msgstr ""
3
  "Project-Id-Version: Cyr To Lat\n"
4
- "POT-Creation-Date: 2021-02-10 14:55+0200\n"
5
- "PO-Revision-Date: 2021-02-10 14:57+0200\n"
6
  "Last-Translator: KAGG Design <info@kagg.eu>\n"
7
  "Language-Team: KAGG Design\n"
8
  "Language: ru_RU\n"
@@ -17,96 +17,219 @@ msgstr ""
17
  "X-Poedit-Flags-xgettext: --add-comments\n"
18
  "X-Poedit-SearchPath-0: .\n"
19
  "X-Poedit-SearchPathExcluded-0: .git\n"
20
- "X-Poedit-SearchPathExcluded-1: .make\n"
21
- "X-Poedit-SearchPathExcluded-2: assets\n"
22
- "X-Poedit-SearchPathExcluded-3: css\n"
23
- "X-Poedit-SearchPathExcluded-4: dist\n"
24
- "X-Poedit-SearchPathExcluded-5: js\n"
25
- "X-Poedit-SearchPathExcluded-6: languages\n"
26
  "X-Poedit-SearchPathExcluded-7: node_modules\n"
27
- "X-Poedit-SearchPathExcluded-8: src\n"
28
  "X-Poedit-SearchPathExcluded-9: tests\n"
29
  "X-Poedit-SearchPathExcluded-10: vendor\n"
30
 
31
- #: classes/background-processes/class-post-conversion-process.php:73
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
32
  msgid "Post slug converted:"
33
  msgstr "Ярлык поста сконвертирован:"
34
 
35
- #: classes/background-processes/class-post-conversion-process.php:102
36
  msgid "Attachment file converted:"
37
  msgstr "Файл вложения сконвертирован:"
38
 
39
- #: classes/background-processes/class-post-conversion-process.php:108
40
  msgid "Cannot convert attachment file for attachment id:"
41
  msgstr "Невозможно сконвертировать файл вложения для вложения с id:"
42
 
43
- #: classes/background-processes/class-post-conversion-process.php:126
44
  msgid "Thumbnail file renamed:"
45
  msgstr "Файл миниатюры переименован:"
46
 
47
- #: classes/background-processes/class-post-conversion-process.php:129
48
  msgid "Cannot rename thumbnail file:"
49
  msgstr "Невозможно переименовать файл миниатюры:"
50
 
51
- #: classes/background-processes/class-post-conversion-process.php:198
52
  msgid "Post slugs conversion completed."
53
  msgstr "Конвертация ярлыков постов завершена."
54
 
55
- #: classes/background-processes/class-term-conversion-process.php:72
56
  msgid "Term slug converted:"
57
  msgstr "Ярлык термина сконвертирован:"
58
 
59
- #: classes/background-processes/class-term-conversion-process.php:84
60
  msgid "Term slugs conversion completed."
61
  msgstr "Конвертация ярлыков терминов завершена."
62
 
63
- #: classes/class-converter.php:106
64
  msgid "Cyr To Lat converts existing post slugs in the background process."
65
  msgstr "Cyr To Lat конвертирует существующие ярлыки постов в фоновом процессе."
66
 
67
- #: classes/class-converter.php:113
68
  msgid "Cyr To Lat converts existing term slugs in the background process."
69
  msgstr ""
70
  "Cyr To Lat конвертирует существующие ярлыки терминов в фоновом процессе."
71
 
72
- #: classes/class-converter.php:120
73
  msgid "Cyr To Lat completed conversion of existing post slugs."
74
  msgstr "Cyr To Lat завершил конвертацию существующих ярлыков постов."
75
 
76
- #: classes/class-converter.php:127
77
  msgid "Cyr To Lat completed conversion of existing term slugs."
78
  msgstr "Cyr To Lat завершил конвертацию существующих ярлыков терминов."
79
 
80
- #: classes/class-converter.php:205
81
  msgid "Cyr To Lat has not found existing post slugs for conversion."
82
  msgstr "Cyr To Lat не нашёл существующих ярлыков постов для конвертации."
83
 
84
- #: classes/class-converter.php:216
85
  msgid "Post slugs conversion started."
86
  msgstr "Начата конвертация ярлыков постов."
87
 
88
- #: classes/class-converter.php:218
89
  msgid "Cyr To Lat started conversion of existing post slugs."
90
  msgstr "Cyr To Lat начал конвертацию существующих ярлыков постов."
91
 
92
- #: classes/class-converter.php:242
93
  msgid "Cyr To Lat has not found existing term slugs for conversion."
94
  msgstr "Cyr To Lat не нашёл существующих ярлыков терминов для конвертации."
95
 
96
- #: classes/class-converter.php:253
97
  msgid "Term slugs conversion started."
98
  msgstr "Начата конвертация ярлыков терминов."
99
 
100
- #: classes/class-converter.php:255
101
  msgid "Cyr To Lat started conversion of existing term slugs."
102
  msgstr "Cyr To Lat начал конвертацию существующих ярлыков терминов."
103
 
104
- #: classes/class-requirements.php:101
105
  msgid "Cyr To Lat plugin has been deactivated."
106
  msgstr "Плагин Cyr To Lat был деактивирован."
107
 
108
  #. translators: 1: Current PHP version number, 2: Cyr To Lat version, 3: Minimum required PHP version number
109
- #: classes/class-requirements.php:116
110
  #, php-format
111
  msgid ""
112
  "Your server is running PHP version %1$s but Cyr To Lat %2$s requires at "
@@ -115,12 +238,12 @@ msgstr ""
115
  "На сервере установлен PHP версии %1$s, однако для Cyr To Lat %2$s требуется "
116
  "хотя бы %3$s."
117
 
118
- #: classes/class-requirements.php:137
119
  msgid "Unable to get filesystem access."
120
  msgstr "Невозможно получить доступ к файловой системе."
121
 
122
  #. translators: 1: max_input_vars value, 2: Cyr To Lat version, 3: Minimum required max_input_vars
123
- #: classes/class-requirements.php:154
124
  #, php-format
125
  msgid ""
126
  "Your server is running PHP with max_input_vars=%1$d but Cyr To Lat %2$s "
@@ -130,166 +253,25 @@ msgstr ""
130
  "требуется хотя бы %3$d."
131
 
132
  #. translators: 1: .user.ini filename
133
- #: classes/class-requirements.php:158
134
  #, php-format
135
  msgid "We have updated settings in %s."
136
  msgstr "Мы обновили настройки в %s."
137
 
138
  #. translators: 1: Wait time in seconds
139
- #: classes/class-requirements.php:161
140
  #, php-format
141
  msgid "Please try again in %d s."
142
  msgstr "Пожалуйста, попробуйте снова через %d сек."
143
 
144
- #: classes/class-requirements.php:226
145
  msgid "Please increase max input vars limit up to 1500."
146
  msgstr "Пожалуйста, увеличьте лимит max input vars до 1500."
147
 
148
- #: classes/class-requirements.php:229
149
  msgid ""
150
  "See: <a href=\"http://sevenspark.com/docs/ubermenu-3/faqs/menu-item-limit\" "
151
  "target=\"_blank\">Increasing max input vars limit.</a>"
152
  msgstr ""
153
  "См.: <a href=\"http://sevenspark.com/docs/ubermenu-3/faqs/menu-item-limit\" "
154
  "target=\"_blank\">Увеличение лимита max input vars.</a>"
155
-
156
- #: classes/class-settings.php:126
157
- msgid "View Cyr To Lat settings"
158
- msgstr "Посмотреть настройки Cyr To Lat"
159
-
160
- #: classes/class-settings.php:127
161
- msgid "Settings"
162
- msgstr "Настройки"
163
-
164
- #: classes/class-settings.php:143
165
- msgid "ISO9 Table"
166
- msgstr "Таблица ISO9"
167
-
168
- #: classes/class-settings.php:146
169
- msgid "bel Table"
170
- msgstr "Таблица bel"
171
-
172
- #: classes/class-settings.php:149
173
- msgid "uk Table"
174
- msgstr "Таблица uk"
175
-
176
- #: classes/class-settings.php:152
177
- msgid "bg_BG Table"
178
- msgstr "Таблица bg_BG"
179
-
180
- #: classes/class-settings.php:155
181
- msgid "mk_MK Table"
182
- msgstr "Таблица mk_MK"
183
-
184
- #: classes/class-settings.php:158
185
- msgid "sr_RS Table"
186
- msgstr "Таблица sr_RS"
187
-
188
- #: classes/class-settings.php:161
189
- msgid "el Table"
190
- msgstr "Таблица el"
191
-
192
- #: classes/class-settings.php:164
193
- msgid "hy Table"
194
- msgstr "Таблица hy"
195
-
196
- #: classes/class-settings.php:167
197
- msgid "ka_GE Table"
198
- msgstr "Таблица ka_GE"
199
-
200
- #: classes/class-settings.php:170
201
- msgid "kk Table"
202
- msgstr "Таблица kk"
203
-
204
- #: classes/class-settings.php:173
205
- msgid "he_IL Table"
206
- msgstr "Таблица he_IL"
207
-
208
- #: classes/class-settings.php:176
209
- msgid "zh_CN Table"
210
- msgstr "Таблица zh_CN"
211
-
212
- #: classes/class-settings.php:203
213
- msgid "(current)"
214
- msgstr "(текущая)"
215
-
216
- #: classes/class-settings.php:270 classes/class-settings.php:271
217
- msgid "Cyr To Lat"
218
- msgstr "Cyr To Lat"
219
-
220
- #. Admin panel title.
221
- #: classes/class-settings.php:291
222
- msgid "Cyr To Lat Plugin Options"
223
- msgstr "Опции плагина Cyr To Lat"
224
-
225
- #: classes/class-settings.php:307
226
- msgid "Convert Existing Slugs"
227
- msgstr "Конвертировать существующие ярлыки"
228
-
229
- #: classes/class-settings.php:313
230
- msgid "Your appreciation"
231
- msgstr "Ваша оценка"
232
-
233
- #: classes/class-settings.php:318
234
- msgid "Leave a ★★★★★ plugin review on WordPress.org"
235
- msgstr "Оставьте ★★★★★ обзор плагина на WordPress.org"
236
-
237
- #: classes/class-settings.php:664
238
- msgid "Options saved."
239
- msgstr "Настройки сохранены."
240
-
241
- #: classes/class-settings.php:665
242
- msgid "Error saving options."
243
- msgstr "Ошибка сохранения настроек."
244
-
245
- #: classes/class-settings.php:689
246
- msgid "Important:"
247
- msgstr "Важно:"
248
-
249
- #: classes/class-settings.php:692
250
- msgid ""
251
- "This operation is irreversible. Please make sure that you have made backup "
252
- "copy of your database."
253
- msgstr ""
254
- "Эта операция необратима. Пожалуйста, убедитесь, что вы сделали копию базы "
255
- "данных."
256
-
257
- #: classes/class-settings.php:697
258
- msgid "Are you sure to continue?"
259
- msgstr "Вы уверены, что хотите продолжить?"
260
-
261
- #: classes/class-settings.php:701
262
- msgid "OK"
263
- msgstr "OK"
264
-
265
- #: classes/class-settings.php:704
266
- msgid "Cancel"
267
- msgstr "Отмена"
268
-
269
- #. translators: %d: cron interval
270
- #: lib/wp-background-processing/class-wp-background-process.php:399
271
- #, php-format
272
- msgid "Every %d Minutes"
273
- msgstr "Каждые %d минут"
274
-
275
- #~ msgid "Donate"
276
- #~ msgstr "Пожертвовать"
277
-
278
- #~ msgid "Would you like to support the advancement of this plugin?"
279
- #~ msgstr "Хотите поддержать развитие этого плагина?"
280
-
281
- #~ msgid "Please try again."
282
- #~ msgstr "Пожалуйста, попробуйте снова."
283
-
284
- #~ msgid "Unable to get filesystem access"
285
- #~ msgstr "Невозможно получить доступ к файловой системе"
286
-
287
- #~ msgid ""
288
- #~ "Your server is running PHP version %1$s but Cyr To Lat version %2$s "
289
- #~ "requires at least %3$s."
290
- #~ msgstr ""
291
- #~ "На сервере установлен PHP версии %1$s, однако для Cyr To Lat %2$s "
292
- #~ "требуется хотя бы %3$s."
293
-
294
- #~ msgid "General Options"
295
- #~ msgstr "Общие настройки"
1
  msgid ""
2
  msgstr ""
3
  "Project-Id-Version: Cyr To Lat\n"
4
+ "POT-Creation-Date: 2021-02-24 22:11+0200\n"
5
+ "PO-Revision-Date: 2021-02-24 22:14+0200\n"
6
  "Last-Translator: KAGG Design <info@kagg.eu>\n"
7
  "Language-Team: KAGG Design\n"
8
  "Language: ru_RU\n"
17
  "X-Poedit-Flags-xgettext: --add-comments\n"
18
  "X-Poedit-SearchPath-0: .\n"
19
  "X-Poedit-SearchPathExcluded-0: .git\n"
20
+ "X-Poedit-SearchPathExcluded-1: .github\n"
21
+ "X-Poedit-SearchPathExcluded-2: .make\n"
22
+ "X-Poedit-SearchPathExcluded-3: .wordpress-org\n"
23
+ "X-Poedit-SearchPathExcluded-4: assets\n"
24
+ "X-Poedit-SearchPathExcluded-5: languages\n"
25
+ "X-Poedit-SearchPathExcluded-6: lib\n"
26
  "X-Poedit-SearchPathExcluded-7: node_modules\n"
27
+ "X-Poedit-SearchPathExcluded-8: src/js\n"
28
  "X-Poedit-SearchPathExcluded-9: tests\n"
29
  "X-Poedit-SearchPathExcluded-10: vendor\n"
30
 
31
+ #: src/php/Settings/Converter.php:59 src/php/Settings/Converter.php:68
32
+ msgid "Converter"
33
+ msgstr "Конвертация"
34
+
35
+ #. Admin panel title.
36
+ #: src/php/Settings/Converter.php:113 src/php/Settings/Tables.php:202
37
+ #: src/php/class-settings.php:291
38
+ msgid "Cyr To Lat Plugin Options"
39
+ msgstr "Опции плагина Cyr To Lat"
40
+
41
+ #: src/php/Settings/Converter.php:128 src/php/class-settings.php:307
42
+ msgid "Convert Existing Slugs"
43
+ msgstr "Конвертировать существующие ярлыки"
44
+
45
+ #: src/php/Settings/Converter.php:161 src/php/class-settings.php:689
46
+ msgid "Important:"
47
+ msgstr "Важно:"
48
+
49
+ #: src/php/Settings/Converter.php:164
50
+ msgid ""
51
+ "This operation is irreversible. Please make sure that you have made a backup "
52
+ "copy of your database."
53
+ msgstr ""
54
+ "Эта операция необратима. Пожалуйста, убедитесь, что вы сделали копию базы "
55
+ "данных."
56
+
57
+ #: src/php/Settings/Converter.php:169 src/php/class-settings.php:697
58
+ msgid "Are you sure to continue?"
59
+ msgstr "Вы уверены, что хотите продолжить?"
60
+
61
+ #: src/php/Settings/Converter.php:173 src/php/class-settings.php:701
62
+ msgid "OK"
63
+ msgstr "OK"
64
+
65
+ #: src/php/Settings/Converter.php:176 src/php/class-settings.php:704
66
+ msgid "Cancel"
67
+ msgstr "Отмена"
68
+
69
+ #: src/php/Settings/PluginSettingsBase.php:52 src/php/class-settings.php:126
70
+ msgid "View Cyr To Lat settings"
71
+ msgstr "Посмотреть настройки Cyr To Lat"
72
+
73
+ #: src/php/Settings/PluginSettingsBase.php:61 src/php/class-settings.php:127
74
+ msgid "Settings"
75
+ msgstr "Настройки"
76
+
77
+ #: src/php/Settings/Tables.php:78
78
+ msgid "Tables"
79
+ msgstr "Таблицы"
80
+
81
+ #: src/php/Settings/Tables.php:87 src/php/class-settings.php:270
82
+ #: src/php/class-settings.php:271
83
+ msgid "Cyr To Lat"
84
+ msgstr "Cyr To Lat"
85
+
86
+ #: src/php/Settings/Tables.php:119 src/php/class-settings.php:143
87
+ msgid "ISO9 Table"
88
+ msgstr "Таблица ISO9"
89
+
90
+ #: src/php/Settings/Tables.php:122 src/php/class-settings.php:146
91
+ msgid "bel Table"
92
+ msgstr "Таблица bel"
93
+
94
+ #: src/php/Settings/Tables.php:125 src/php/class-settings.php:149
95
+ msgid "uk Table"
96
+ msgstr "Таблица uk"
97
+
98
+ #: src/php/Settings/Tables.php:128 src/php/class-settings.php:152
99
+ msgid "bg_BG Table"
100
+ msgstr "Таблица bg_BG"
101
+
102
+ #: src/php/Settings/Tables.php:131 src/php/class-settings.php:155
103
+ msgid "mk_MK Table"
104
+ msgstr "Таблица mk_MK"
105
+
106
+ #: src/php/Settings/Tables.php:134 src/php/class-settings.php:158
107
+ msgid "sr_RS Table"
108
+ msgstr "Таблица sr_RS"
109
+
110
+ #: src/php/Settings/Tables.php:137 src/php/class-settings.php:161
111
+ msgid "el Table"
112
+ msgstr "Таблица el"
113
+
114
+ #: src/php/Settings/Tables.php:140 src/php/class-settings.php:164
115
+ msgid "hy Table"
116
+ msgstr "Таблица hy"
117
+
118
+ #: src/php/Settings/Tables.php:143 src/php/class-settings.php:167
119
+ msgid "ka_GE Table"
120
+ msgstr "Таблица ka_GE"
121
+
122
+ #: src/php/Settings/Tables.php:146 src/php/class-settings.php:170
123
+ msgid "kk Table"
124
+ msgstr "Таблица kk"
125
+
126
+ #: src/php/Settings/Tables.php:149 src/php/class-settings.php:173
127
+ msgid "he_IL Table"
128
+ msgstr "Таблица he_IL"
129
+
130
+ #: src/php/Settings/Tables.php:152 src/php/class-settings.php:176
131
+ msgid "zh_CN Table"
132
+ msgstr "Таблица zh_CN"
133
+
134
+ #: src/php/Settings/Tables.php:179 src/php/class-settings.php:203
135
+ msgid "(current)"
136
+ msgstr "(текущая)"
137
+
138
+ #: src/php/Settings/Tables.php:216 src/php/class-settings.php:313
139
+ msgid "Your appreciation"
140
+ msgstr "Ваша оценка"
141
+
142
+ #: src/php/Settings/Tables.php:221 src/php/class-settings.php:318
143
+ msgid "Leave a ★★★★★ plugin review on WordPress.org"
144
+ msgstr "Оставьте ★★★★★ обзор плагина на WordPress.org"
145
+
146
+ #: src/php/Settings/Tables.php:260 src/php/class-settings.php:664
147
+ msgid "Options saved."
148
+ msgstr "Настройки сохранены."
149
+
150
+ #: src/php/Settings/Tables.php:261 src/php/class-settings.php:665
151
+ msgid "Error saving options."
152
+ msgstr "Ошибка сохранения настроек."
153
+
154
+ #: src/php/background-processes/class-post-conversion-process.php:73
155
  msgid "Post slug converted:"
156
  msgstr "Ярлык поста сконвертирован:"
157
 
158
+ #: src/php/background-processes/class-post-conversion-process.php:102
159
  msgid "Attachment file converted:"
160
  msgstr "Файл вложения сконвертирован:"
161
 
162
+ #: src/php/background-processes/class-post-conversion-process.php:108
163
  msgid "Cannot convert attachment file for attachment id:"
164
  msgstr "Невозможно сконвертировать файл вложения для вложения с id:"
165
 
166
+ #: src/php/background-processes/class-post-conversion-process.php:126
167
  msgid "Thumbnail file renamed:"
168
  msgstr "Файл миниатюры переименован:"
169
 
170
+ #: src/php/background-processes/class-post-conversion-process.php:129
171
  msgid "Cannot rename thumbnail file:"
172
  msgstr "Невозможно переименовать файл миниатюры:"
173
 
174
+ #: src/php/background-processes/class-post-conversion-process.php:198
175
  msgid "Post slugs conversion completed."
176
  msgstr "Конвертация ярлыков постов завершена."
177
 
178
+ #: src/php/background-processes/class-term-conversion-process.php:72
179
  msgid "Term slug converted:"
180
  msgstr "Ярлык термина сконвертирован:"
181
 
182
+ #: src/php/background-processes/class-term-conversion-process.php:84
183
  msgid "Term slugs conversion completed."
184
  msgstr "Конвертация ярлыков терминов завершена."
185
 
186
+ #: src/php/class-converter.php:106
187
  msgid "Cyr To Lat converts existing post slugs in the background process."
188
  msgstr "Cyr To Lat конвертирует существующие ярлыки постов в фоновом процессе."
189
 
190
+ #: src/php/class-converter.php:113
191
  msgid "Cyr To Lat converts existing term slugs in the background process."
192
  msgstr ""
193
  "Cyr To Lat конвертирует существующие ярлыки терминов в фоновом процессе."
194
 
195
+ #: src/php/class-converter.php:120
196
  msgid "Cyr To Lat completed conversion of existing post slugs."
197
  msgstr "Cyr To Lat завершил конвертацию существующих ярлыков постов."
198
 
199
+ #: src/php/class-converter.php:127
200
  msgid "Cyr To Lat completed conversion of existing term slugs."
201
  msgstr "Cyr To Lat завершил конвертацию существующих ярлыков терминов."
202
 
203
+ #: src/php/class-converter.php:208
204
  msgid "Cyr To Lat has not found existing post slugs for conversion."
205
  msgstr "Cyr To Lat не нашёл существующих ярлыков постов для конвертации."
206
 
207
+ #: src/php/class-converter.php:219
208
  msgid "Post slugs conversion started."
209
  msgstr "Начата конвертация ярлыков постов."
210
 
211
+ #: src/php/class-converter.php:221
212
  msgid "Cyr To Lat started conversion of existing post slugs."
213
  msgstr "Cyr To Lat начал конвертацию существующих ярлыков постов."
214
 
215
+ #: src/php/class-converter.php:245
216
  msgid "Cyr To Lat has not found existing term slugs for conversion."
217
  msgstr "Cyr To Lat не нашёл существующих ярлыков терминов для конвертации."
218
 
219
+ #: src/php/class-converter.php:256
220
  msgid "Term slugs conversion started."
221
  msgstr "Начата конвертация ярлыков терминов."
222
 
223
+ #: src/php/class-converter.php:258
224
  msgid "Cyr To Lat started conversion of existing term slugs."
225
  msgstr "Cyr To Lat начал конвертацию существующих ярлыков терминов."
226
 
227
+ #: src/php/class-requirements.php:101
228
  msgid "Cyr To Lat plugin has been deactivated."
229
  msgstr "Плагин Cyr To Lat был деактивирован."
230
 
231
  #. translators: 1: Current PHP version number, 2: Cyr To Lat version, 3: Minimum required PHP version number
232
+ #: src/php/class-requirements.php:116
233
  #, php-format
234
  msgid ""
235
  "Your server is running PHP version %1$s but Cyr To Lat %2$s requires at "
238
  "На сервере установлен PHP версии %1$s, однако для Cyr To Lat %2$s требуется "
239
  "хотя бы %3$s."
240
 
241
+ #: src/php/class-requirements.php:137
242
  msgid "Unable to get filesystem access."
243
  msgstr "Невозможно получить доступ к файловой системе."
244
 
245
  #. translators: 1: max_input_vars value, 2: Cyr To Lat version, 3: Minimum required max_input_vars
246
+ #: src/php/class-requirements.php:154
247
  #, php-format
248
  msgid ""
249
  "Your server is running PHP with max_input_vars=%1$d but Cyr To Lat %2$s "
253
  "требуется хотя бы %3$d."
254
 
255
  #. translators: 1: .user.ini filename
256
+ #: src/php/class-requirements.php:158
257
  #, php-format
258
  msgid "We have updated settings in %s."
259
  msgstr "Мы обновили настройки в %s."
260
 
261
  #. translators: 1: Wait time in seconds
262
+ #: src/php/class-requirements.php:161
263
  #, php-format
264
  msgid "Please try again in %d s."
265
  msgstr "Пожалуйста, попробуйте снова через %d сек."
266
 
267
+ #: src/php/class-requirements.php:226
268
  msgid "Please increase max input vars limit up to 1500."
269
  msgstr "Пожалуйста, увеличьте лимит max input vars до 1500."
270
 
271
+ #: src/php/class-requirements.php:229
272
  msgid ""
273
  "See: <a href=\"http://sevenspark.com/docs/ubermenu-3/faqs/menu-item-limit\" "
274
  "target=\"_blank\">Increasing max input vars limit.</a>"
275
  msgstr ""
276
  "См.: <a href=\"http://sevenspark.com/docs/ubermenu-3/faqs/menu-item-limit\" "
277
  "target=\"_blank\">Увеличение лимита max input vars.</a>"
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
languages/cyr2lat-sv.po CHANGED
@@ -16,14 +16,14 @@ msgstr ""
16
  "X-Poedit-Flags-xgettext: --add-comments\n"
17
  "X-Poedit-SearchPath-0: .\n"
18
  "X-Poedit-SearchPathExcluded-0: .git\n"
19
- "X-Poedit-SearchPathExcluded-1: .make\n"
20
- "X-Poedit-SearchPathExcluded-2: assets\n"
21
- "X-Poedit-SearchPathExcluded-3: css\n"
22
- "X-Poedit-SearchPathExcluded-4: dist\n"
23
- "X-Poedit-SearchPathExcluded-5: js\n"
24
- "X-Poedit-SearchPathExcluded-6: languages\n"
25
  "X-Poedit-SearchPathExcluded-7: node_modules\n"
26
- "X-Poedit-SearchPathExcluded-8: src\n"
27
  "X-Poedit-SearchPathExcluded-9: tests\n"
28
  "X-Poedit-SearchPathExcluded-10: vendor\n"
29
 
16
  "X-Poedit-Flags-xgettext: --add-comments\n"
17
  "X-Poedit-SearchPath-0: .\n"
18
  "X-Poedit-SearchPathExcluded-0: .git\n"
19
+ "X-Poedit-SearchPathExcluded-1: .github\n"
20
+ "X-Poedit-SearchPathExcluded-2: .make\n"
21
+ "X-Poedit-SearchPathExcluded-3: .wordpress-org\n"
22
+ "X-Poedit-SearchPathExcluded-4: assets\n"
23
+ "X-Poedit-SearchPathExcluded-5: languages\n"
24
+ "X-Poedit-SearchPathExcluded-6: lib\n"
25
  "X-Poedit-SearchPathExcluded-7: node_modules\n"
26
+ "X-Poedit-SearchPathExcluded-8: src/js\n"
27
  "X-Poedit-SearchPathExcluded-9: tests\n"
28
  "X-Poedit-SearchPathExcluded-10: vendor\n"
29
 
languages/cyr2lat-uk.po CHANGED
@@ -17,14 +17,14 @@ msgstr ""
17
  "X-Poedit-Flags-xgettext: --add-comments\n"
18
  "X-Poedit-SearchPath-0: .\n"
19
  "X-Poedit-SearchPathExcluded-0: .git\n"
20
- "X-Poedit-SearchPathExcluded-1: .make\n"
21
- "X-Poedit-SearchPathExcluded-2: assets\n"
22
- "X-Poedit-SearchPathExcluded-3: css\n"
23
- "X-Poedit-SearchPathExcluded-4: dist\n"
24
- "X-Poedit-SearchPathExcluded-5: js\n"
25
- "X-Poedit-SearchPathExcluded-6: languages\n"
26
  "X-Poedit-SearchPathExcluded-7: node_modules\n"
27
- "X-Poedit-SearchPathExcluded-8: src\n"
28
  "X-Poedit-SearchPathExcluded-9: tests\n"
29
  "X-Poedit-SearchPathExcluded-10: vendor\n"
30
 
17
  "X-Poedit-Flags-xgettext: --add-comments\n"
18
  "X-Poedit-SearchPath-0: .\n"
19
  "X-Poedit-SearchPathExcluded-0: .git\n"
20
+ "X-Poedit-SearchPathExcluded-1: .github\n"
21
+ "X-Poedit-SearchPathExcluded-2: .make\n"
22
+ "X-Poedit-SearchPathExcluded-3: .wordpress-org\n"
23
+ "X-Poedit-SearchPathExcluded-4: assets\n"
24
+ "X-Poedit-SearchPathExcluded-5: languages\n"
25
+ "X-Poedit-SearchPathExcluded-6: lib\n"
26
  "X-Poedit-SearchPathExcluded-7: node_modules\n"
27
+ "X-Poedit-SearchPathExcluded-8: src/js\n"
28
  "X-Poedit-SearchPathExcluded-9: tests\n"
29
  "X-Poedit-SearchPathExcluded-10: vendor\n"
30
 
readme.txt CHANGED
@@ -3,7 +3,7 @@ Contributors: SergeyBiryukov, mihdan, karevn, webvitaly, kaggdesign
3
  Tags: cyrillic, belorussian, ukrainian, bulgarian, macedonian, georgian, kazakh, latin, l10n, russian, cyr-to-lat, cyr2lat, rustolat, slugs, translations, transliteration
4
  Requires at least: 5.1
5
  Tested up to: 5.7
6
- Stable tag: 4.6.4
7
  Requires PHP: 5.6.20
8
 
9
  Converts Cyrillic characters in post, page and term slugs to Latin characters.
@@ -21,11 +21,15 @@ Converts Cyrillic characters in post, page and term slugs to Latin characters. U
21
  * [Has many advantages over similar plugins](https://kagg.eu/en/the-benefits-of-cyr-to-lat/)
22
  * [Officially compatible with WPML](https://wpml.org/plugin/cyr-to-lat/)
23
 
24
- ![WPML Certificate](https://ps.w.org/cyr2lat/assets/Cyr-To-Lat---WPML-Compatibility-Certificate-240x250.png)
25
 
26
  Based on the original Rus-To-Lat plugin by Anton Skorobogatov.
27
 
28
- [](http://coderisk.com/wp/plugin/cyr2lat/RIPS-nt7iXCmzoc)
 
 
 
 
29
 
30
  == Installation ==
31
 
@@ -174,6 +178,12 @@ Yes you can!
174
 
175
  == Changelog ==
176
 
 
 
 
 
 
 
177
  = 4.6.4 (03.03.2021) =
178
  * Tested up to WordPress 5.7
179
 
3
  Tags: cyrillic, belorussian, ukrainian, bulgarian, macedonian, georgian, kazakh, latin, l10n, russian, cyr-to-lat, cyr2lat, rustolat, slugs, translations, transliteration
4
  Requires at least: 5.1
5
  Tested up to: 5.7
6
+ Stable tag: 5.0.0
7
  Requires PHP: 5.6.20
8
 
9
  Converts Cyrillic characters in post, page and term slugs to Latin characters.
21
  * [Has many advantages over similar plugins](https://kagg.eu/en/the-benefits-of-cyr-to-lat/)
22
  * [Officially compatible with WPML](https://wpml.org/plugin/cyr-to-lat/)
23
 
24
+ <img src="https://ps.w.org/cyr2lat/assets/Cyr-To-Lat---WPML-Compatibility-Certificate-240x250.png" alt="WPML Certificate" />
25
 
26
  Based on the original Rus-To-Lat plugin by Anton Skorobogatov.
27
 
28
+ == Screenshots ==
29
+
30
+ 1. Tables settings page
31
+ 2. Converter settings page
32
+ 3. Block editor with transliterated slug
33
 
34
  == Installation ==
35
 
178
 
179
  == Changelog ==
180
 
181
+ = 5.0.0 (18.03.2021) =
182
+ * Introduce tabs on options page
183
+ * Add options to select post types and statuses for background conversion
184
+ * Make colors compatible to WP official palette
185
+ * Fix bug with Polylang when locale is not equal to language slug
186
+
187
  = 4.6.4 (03.03.2021) =
188
  * Tested up to WordPress 5.7
189
 
src/php/Settings/Abstracts/SettingsBase.php ADDED
@@ -0,0 +1,956 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ /**
3
+ * SettingsBase class file.
4
+ *
5
+ * @package cyr-to-lat
6
+ */
7
+
8
+ namespace Cyr_To_Lat\Settings\Abstracts;
9
+
10
+ /**
11
+ * Class SettingsBase
12
+ *
13
+ * This is an abstract class to create the settings page in any plugin.
14
+ * It uses WordPress Settings API and general output of fields of any type.
15
+ * Similar approach is used in many plugins, including WooCommerce.
16
+ */
17
+ abstract class SettingsBase {
18
+
19
+ /**
20
+ * Admin script handle.
21
+ */
22
+ const HANDLE = 'ctl-settings-base';
23
+
24
+ /**
25
+ * Form fields.
26
+ *
27
+ * @var array
28
+ */
29
+ protected $form_fields;
30
+
31
+ /**
32
+ * Plugin options.
33
+ *
34
+ * @var array
35
+ */
36
+ protected $settings;
37
+
38
+ /**
39
+ * Tabs of this settings page.
40
+ *
41
+ * @var array|null
42
+ */
43
+ protected $tabs;
44
+
45
+ /**
46
+ * Get screen id.
47
+ *
48
+ * @return string
49
+ */
50
+ abstract public function screen_id();
51
+
52
+ /**
53
+ * Get option group.
54
+ *
55
+ * @return string
56
+ */
57
+ abstract protected function option_group();
58
+
59
+ /**
60
+ * Get option page.
61
+ *
62
+ * @return string
63
+ */
64
+ abstract protected function option_page();
65
+
66
+ /**
67
+ * Get option name.
68
+ *
69
+ * @return string
70
+ */
71
+ abstract protected function option_name();
72
+
73
+ /**
74
+ * Get plugin base name.
75
+ *
76
+ * @return string
77
+ */
78
+ abstract protected function plugin_basename();
79
+
80
+ /**
81
+ * Get plugin url.
82
+ *
83
+ * @return string
84
+ */
85
+ abstract protected function plugin_url();
86
+
87
+ /**
88
+ * Get plugin version.
89
+ *
90
+ * @return string
91
+ */
92
+ abstract protected function plugin_version();
93
+
94
+ /**
95
+ * Get settings link label.
96
+ *
97
+ * @return string
98
+ */
99
+ abstract protected function settings_link_label();
100
+
101
+ /**
102
+ * Get settings link text.
103
+ *
104
+ * @return string
105
+ */
106
+ abstract protected function settings_link_text();
107
+
108
+ /**
109
+ * Init form fields.
110
+ */
111
+ abstract protected function init_form_fields();
112
+
113
+ /**
114
+ * Get page title.
115
+ *
116
+ * @return string
117
+ */
118
+ abstract protected function page_title();
119
+
120
+ /**
121
+ * Get menu title.
122
+ *
123
+ * @return string
124
+ */
125
+ abstract protected function menu_title();
126
+
127
+ /**
128
+ * Show setting page.
129
+ */
130
+ abstract public function settings_page();
131
+
132
+ /**
133
+ * Get section title.
134
+ *
135
+ * @return string
136
+ */
137
+ abstract protected function section_title();
138
+
139
+ /**
140
+ * Show section.
141
+ *
142
+ * @param array $arguments Arguments.
143
+ */
144
+ abstract public function section_callback( $arguments );
145
+
146
+ /**
147
+ * Enqueue scripts in admin.
148
+ */
149
+ abstract public function admin_enqueue_scripts();
150
+
151
+ /**
152
+ * Get text domain.
153
+ *
154
+ * @return string
155
+ */
156
+ abstract protected function text_domain();
157
+
158
+ /**
159
+ * SettingsBase constructor.
160
+ *
161
+ * @param array $tabs Tabs of this settings page.
162
+ */
163
+ public function __construct( $tabs = [] ) {
164
+ $this->tabs = $tabs;
165
+
166
+ if ( ! $this->is_tab() ) {
167
+ add_action( 'current_screen', [ $this, 'setup_tabs_section' ] );
168
+ }
169
+
170
+ if ( $this->is_tab_active( $this ) ) {
171
+ add_action( 'plugins_loaded', [ $this, 'init' ] );
172
+ }
173
+ }
174
+
175
+ /**
176
+ * Init class.
177
+ */
178
+ public function init() {
179
+ $this->load_plugin_textdomain();
180
+ $this->init_form_fields();
181
+ $this->init_settings();
182
+ $this->init_hooks();
183
+ }
184
+
185
+ /**
186
+ * Init class hooks.
187
+ */
188
+ protected function init_hooks() {
189
+ add_filter(
190
+ 'plugin_action_links_' . $this->plugin_basename(),
191
+ [ $this, 'add_settings_link' ],
192
+ 10
193
+ );
194
+
195
+ add_action( 'admin_menu', [ $this, 'add_settings_page' ] );
196
+ add_action( 'current_screen', [ $this, 'setup_sections' ] );
197
+ add_action( 'current_screen', [ $this, 'setup_fields' ] );
198
+
199
+ add_filter( 'pre_update_option_' . $this->option_name(), [ $this, 'pre_update_option_filter' ], 10, 2 );
200
+
201
+ add_action( 'admin_enqueue_scripts', [ $this, 'base_admin_enqueue_scripts' ] );
202
+ }
203
+
204
+ /**
205
+ * Get parent slug.
206
+ *
207
+ * @return string
208
+ */
209
+ protected function parent_slug() {
210
+ // By default, add menu pages to Options menu.
211
+ return 'options-general.php';
212
+ }
213
+
214
+ /**
215
+ * Is this the main menu page.
216
+ *
217
+ * @return bool
218
+ */
219
+ protected function is_main_menu_page() {
220
+ // Main menu page should have empty string as parent slug.
221
+ return ! (bool) $this->parent_slug();
222
+ }
223
+
224
+ /**
225
+ * Get tab name.
226
+ *
227
+ * @return string
228
+ */
229
+ protected function tab_name() {
230
+ return $this->get_class_name();
231
+ }
232
+
233
+ /**
234
+ * Get class name without namespace.
235
+ *
236
+ * @return string
237
+ */
238
+ protected function get_class_name() {
239
+ $path = explode( '\\', get_class( $this ) );
240
+
241
+ return array_pop( $path );
242
+ }
243
+
244
+ /**
245
+ * Is this a tab.
246
+ *
247
+ * @return bool
248
+ */
249
+ protected function is_tab() {
250
+ // Tab has null in tabs property.
251
+ return null === $this->tabs;
252
+ }
253
+
254
+ /**
255
+ * Add link to plugin setting page on plugins page.
256
+ *
257
+ * @param array $actions An array of plugin action links.
258
+ * By default this can include 'activate', 'deactivate', and 'delete'.
259
+ * With Multisite active this can also include
260
+ * 'network_active' and 'network_only' items.
261
+ *
262
+ * @return array|mixed Plugin links
263
+ */
264
+ public function add_settings_link( array $actions ) {
265
+ $new_actions = [
266
+ 'settings' =>
267
+ '<a href="' . admin_url( 'options-general.php?page=' . $this->option_page() ) .
268
+ '" aria-label="' . esc_attr( $this->settings_link_label() ) . '">' .
269
+ esc_html( $this->settings_link_text() ) . '</a>',
270
+ ];
271
+
272
+ return array_merge( $new_actions, $actions );
273
+ }
274
+
275
+ /**
276
+ * Initialise Settings.
277
+ *
278
+ * Store all settings in a single database entry
279
+ * and make sure the $settings array is either the default
280
+ * or the settings stored in the database.
281
+ */
282
+ protected function init_settings() {
283
+ $this->settings = get_option( $this->option_name(), null );
284
+
285
+ $form_fields = $this->form_fields();
286
+
287
+ if ( is_array( $this->settings ) ) {
288
+ $this->settings = array_merge( wp_list_pluck( $form_fields, 'default' ), $this->settings );
289
+
290
+ return;
291
+ }
292
+
293
+ // If there are no settings defined, use defaults.
294
+ $this->settings = array_merge(
295
+ array_fill_keys( array_keys( $form_fields ), '' ),
296
+ wp_list_pluck( $form_fields, 'default' )
297
+ );
298
+ }
299
+
300
+ /**
301
+ * Get the form fields after initialization.
302
+ *
303
+ * @return array of options
304
+ */
305
+ protected function form_fields() {
306
+ if ( empty( $this->form_fields ) ) {
307
+ $this->init_form_fields();
308
+ }
309
+
310
+ return array_map( [ $this, 'set_defaults' ], $this->form_fields );
311
+ }
312
+
313
+ /**
314
+ * Set default required properties for each field.
315
+ *
316
+ * @param array $field Settings field.
317
+ *
318
+ * @return array
319
+ */
320
+ protected function set_defaults( $field ) {
321
+ if ( ! isset( $field['default'] ) ) {
322
+ $field['default'] = '';
323
+ }
324
+
325
+ return $field;
326
+ }
327
+
328
+ /**
329
+ * Add settings page to the menu.
330
+ */
331
+ public function add_settings_page() {
332
+ if ( $this->is_main_menu_page() ) {
333
+ add_menu_page(
334
+ $this->page_title(),
335
+ $this->menu_title(),
336
+ 'manage_options',
337
+ $this->option_page(),
338
+ [ $this, 'settings_base_page' ]
339
+ );
340
+
341
+ return;
342
+ }
343
+
344
+ add_submenu_page(
345
+ $this->parent_slug(),
346
+ $this->page_title(),
347
+ $this->menu_title(),
348
+ 'manage_options',
349
+ $this->option_page(),
350
+ [ $this, 'settings_base_page' ]
351
+ );
352
+ }
353
+
354
+ /**
355
+ * Invoke relevant settings_page() basing on tabs.
356
+ */
357
+ public function settings_base_page() {
358
+ $this->get_active_tab()->settings_page();
359
+ }
360
+
361
+ /**
362
+ * Invoke relevant admin_enqueue_scripts() basing on tabs.
363
+ */
364
+ public function base_admin_enqueue_scripts() {
365
+ $this->get_active_tab()->admin_enqueue_scripts();
366
+
367
+ wp_enqueue_style(
368
+ self::HANDLE,
369
+ $this->plugin_url() . '/assets/css/settings-base.css',
370
+ [],
371
+ $this->plugin_version()
372
+ );
373
+ }
374
+
375
+ /**
376
+ * Setup settings sections.
377
+ */
378
+ public function setup_sections() {
379
+ $tab = $this->get_active_tab();
380
+
381
+ foreach ( $this->form_fields as $form_field ) {
382
+ $title = isset( $form_field['title'] ) ? $form_field['title'] : '';
383
+ add_settings_section(
384
+ $form_field['section'],
385
+ $title,
386
+ [ $tab, 'section_callback' ],
387
+ $tab->option_page()
388
+ );
389
+ }
390
+ }
391
+
392
+ /**
393
+ * Setup tabs section.
394
+ */
395
+ public function setup_tabs_section() {
396
+ $tab = $this->get_active_tab();
397
+
398
+ add_settings_section(
399
+ 'tabs_section',
400
+ '',
401
+ [ $this, 'tabs_callback' ],
402
+ $tab->option_page()
403
+ );
404
+ }
405
+
406
+ /**
407
+ * Show tabs.
408
+ */
409
+ public function tabs_callback() {
410
+ ?>
411
+ <div class="ctl-settings-tabs">
412
+ <?php
413
+ $this->tab_link( $this );
414
+
415
+ foreach ( $this->tabs as $tab ) {
416
+ $this->tab_link( $tab );
417
+ }
418
+ ?>
419
+ </div>
420
+ <?php
421
+ }
422
+
423
+ /**
424
+ * Show tab link.
425
+ *
426
+ * @param SettingsBase $tab Tabs of the current settings page.
427
+ */
428
+ private function tab_link( $tab ) {
429
+ $url = menu_page_url( $this->option_page(), false );
430
+ $url = add_query_arg( 'tab', strtolower( $tab->get_class_name() ), $url );
431
+ $active = $this->is_tab_active( $tab ) ? ' active' : '';
432
+
433
+ ?>
434
+ <a class="ctl-settings-tab<?php echo esc_attr( $active ); ?>" href="<?php echo esc_url( $url ); ?>">
435
+ <?php echo esc_html( $tab->tab_name() ); ?>
436
+ </a>
437
+ <?php
438
+ }
439
+
440
+ /**
441
+ * Check if tab is active.
442
+ *
443
+ * @param SettingsBase $tab Tab of the current settings page.
444
+ *
445
+ * @return bool
446
+ */
447
+ protected function is_tab_active( $tab ) {
448
+ $current_tab_name = filter_input( INPUT_GET, 'tab', FILTER_SANITIZE_STRING );
449
+
450
+ if ( null === $current_tab_name && ! $tab->is_tab() ) {
451
+ return true;
452
+ }
453
+
454
+ return strtolower( $tab->get_class_name() ) === $current_tab_name;
455
+ }
456
+
457
+ /**
458
+ * Get active tab.
459
+ *
460
+ * @return SettingsBase
461
+ */
462
+ protected function get_active_tab() {
463
+ if ( ! empty( $this->tabs ) ) {
464
+ foreach ( $this->tabs as $tab ) {
465
+ if ( $this->is_tab_active( $tab ) ) {
466
+ return $tab;
467
+ }
468
+ }
469
+ }
470
+
471
+ return $this;
472
+ }
473
+
474
+ /**
475
+ * Setup settings fields.
476
+ */
477
+ public function setup_fields() {
478
+ if ( ! $this->is_options_screen() ) {
479
+ return;
480
+ }
481
+
482
+ register_setting( $this->option_group(), $this->option_name() );
483
+
484
+ foreach ( $this->form_fields as $key => $field ) {
485
+ $field['field_id'] = $key;
486
+
487
+ add_settings_field(
488
+ $key,
489
+ $field['label'],
490
+ [ $this, 'field_callback' ],
491
+ $this->option_page(),
492
+ $field['section'],
493
+ $field
494
+ );
495
+ }
496
+ }
497
+
498
+ /**
499
+ * Print text/password field.
500
+ *
501
+ * @param array $arguments Field arguments.
502
+ *
503
+ * @noinspection PhpUnusedPrivateMethodInspection
504
+ */
505
+ private function print_text_field( array $arguments ) {
506
+ $value = $this->get( $arguments['field_id'] );
507
+
508
+ printf(
509
+ '<input name="%1$s[%2$s]" id="%2$s" type="%3$s"' .
510
+ ' placeholder="%4$s" value="%5$s" class="regular-text" />',
511
+ esc_html( $this->option_name() ),
512
+ esc_attr( $arguments['field_id'] ),
513
+ esc_attr( $arguments['type'] ),
514
+ esc_attr( $arguments['placeholder'] ),
515
+ esc_html( $value )
516
+ );
517
+ }
518
+
519
+ /**
520
+ * Print number field.
521
+ *
522
+ * @param array $arguments Field arguments.
523
+ *
524
+ * @noinspection PhpUnusedPrivateMethodInspection
525
+ */
526
+ private function print_number_field( array $arguments ) {
527
+ $value = $this->get( $arguments['field_id'] );
528
+ $min = isset( $arguments['min'] ) ? $arguments['min'] : '';
529
+ $max = isset( $arguments['max'] ) ? $arguments['max'] : '';
530
+
531
+ printf(
532
+ '<input name="%1$s[%2$s]" id="%2$s" type="%3$s"' .
533
+ ' placeholder="%4$s" value="%5$s" class="regular-text" min="%6$s" max="%7$s" />',
534
+ esc_html( $this->option_name() ),
535
+ esc_attr( $arguments['field_id'] ),
536
+ esc_attr( $arguments['type'] ),
537
+ esc_attr( $arguments['placeholder'] ),
538
+ esc_html( $value ),
539
+ esc_attr( $min ),
540
+ esc_attr( $max )
541
+ );
542
+ }
543
+
544
+ /**
545
+ * Print textarea field.
546
+ *
547
+ * @param array $arguments Field arguments.
548
+ *
549
+ * @noinspection PhpUnusedPrivateMethodInspection
550
+ */
551
+ private function print_text_area_field( array $arguments ) {
552
+ $value = $this->get( $arguments['field_id'] );
553
+
554
+ printf(
555
+ '<textarea name="%1$s[%2$s]" id="%2$s" placeholder="%3$s" rows="5" cols="50">%4$s</textarea>',
556
+ esc_html( $this->option_name() ),
557
+ esc_attr( $arguments['field_id'] ),
558
+ esc_attr( $arguments['placeholder'] ),
559
+ wp_kses_post( $value )
560
+ );
561
+ }
562
+
563
+ /**
564
+ * Print checkbox field.
565
+ *
566
+ * @param array $arguments Field arguments.
567
+ *
568
+ * @noinspection PhpUnusedPrivateMethodInspection
569
+ */
570
+ private function print_check_box_field( array $arguments ) {
571
+ $value = (array) $this->get( $arguments['field_id'] );
572
+
573
+ if ( empty( $arguments['options'] ) || ! is_array( $arguments['options'] ) ) {
574
+ $arguments['options'] = [ 'yes' => '' ];
575
+ }
576
+
577
+ $options_markup = '';
578
+ $iterator = 0;
579
+ foreach ( $arguments['options'] as $key => $label ) {
580
+ $iterator ++;
581
+ $checked = false;
582
+ if ( is_array( $value ) && in_array( $key, $value, true ) ) {
583
+ $checked = checked( $key, $key, false );
584
+ }
585
+ $options_markup .= sprintf(
586
+ '<label for="%2$s_%7$s">' .
587
+ '<input id="%2$s_%7$s" name="%1$s[%2$s][]" type="%3$s" value="%4$s" %5$s />' .
588
+ ' %6$s' .
589
+ '</label>' .
590
+ '<br/>',
591
+ esc_html( $this->option_name() ),
592
+ $arguments['field_id'],
593
+ $arguments['type'],
594
+ $key,
595
+ $checked,
596
+ $label,
597
+ $iterator
598
+ );
599
+ }
600
+
601
+ printf(
602
+ '<fieldset>%s</fieldset>',
603
+ wp_kses(
604
+ $options_markup,
605
+ [
606
+ 'label' => [
607
+ 'for' => [],
608
+ ],
609
+ 'input' => [
610
+ 'id' => [],
611
+ 'name' => [],
612
+ 'type' => [],
613
+ 'value' => [],
614
+ 'checked' => [],
615
+ ],
616
+ 'br' => [],
617
+ ]
618
+ )
619
+ );
620
+ }
621
+
622
+ /**
623
+ * Print radio field.
624
+ *
625
+ * @param array $arguments Field arguments.
626
+ *
627
+ * @noinspection PhpUnusedPrivateMethodInspection
628
+ */
629
+ private function print_radio_field( array $arguments ) {
630
+ $value = $this->get( $arguments['field_id'] );
631
+
632
+ if ( empty( $arguments['options'] ) || ! is_array( $arguments['options'] ) ) {
633
+ return;
634
+ }
635
+
636
+ $options_markup = '';
637
+ $iterator = 0;
638
+ foreach ( $arguments['options'] as $key => $label ) {
639
+ $iterator ++;
640
+ $options_markup .= sprintf(
641
+ '<label for="%2$s_%7$s">' .
642
+ '<input id="%2$s_%7$s" name="%1$s[%2$s]" type="%3$s" value="%4$s" %5$s />' .
643
+ ' %6$s' .
644
+ '</label>' .
645
+ '<br/>',
646
+ esc_html( $this->option_name() ),
647
+ $arguments['field_id'],
648
+ $arguments['type'],
649
+ $key,
650
+ checked( $value, $key, false ),
651
+ $label,
652
+ $iterator
653
+ );
654
+ }
655
+
656
+ printf(
657
+ '<fieldset>%s</fieldset>',
658
+ wp_kses(
659
+ $options_markup,
660
+ [
661
+ 'label' => [
662
+ 'for' => [],
663
+ ],
664
+ 'input' => [
665
+ 'id' => [],
666
+ 'name' => [],
667
+ 'type' => [],
668
+ 'value' => [],
669
+ 'checked' => [],
670
+ ],
671
+ 'br' => [],
672
+ ]
673
+ )
674
+ );
675
+ }
676
+
677
+ /**
678
+ * Print select field.
679
+ *
680
+ * @param array $arguments Field arguments.
681
+ *
682
+ * @noinspection PhpUnusedPrivateMethodInspection
683
+ */
684
+ private function print_select_field( array $arguments ) {
685
+ $value = $this->get( $arguments['field_id'] );
686
+
687
+ if ( empty( $arguments['options'] ) || ! is_array( $arguments['options'] ) ) {
688
+ return;
689
+ }
690
+
691
+ $options_markup = '';
692
+ foreach ( $arguments['options'] as $key => $label ) {
693
+ $options_markup .= sprintf(
694
+ '<option value="%s" %s>%s</option>',
695
+ $key,
696
+ selected( $value, $key, false ),
697
+ $label
698
+ );
699
+ }
700
+
701
+ printf(
702
+ '<select name="%1$s[%2$s]">%3$s</select>',
703
+ esc_html( $this->option_name() ),
704
+ esc_html( $arguments['field_id'] ),
705
+ wp_kses(
706
+ $options_markup,
707
+ [
708
+ 'option' => [
709
+ 'value' => [],
710
+ 'selected' => [],
711
+ ],
712
+ ]
713
+ )
714
+ );
715
+ }
716
+
717
+ /**
718
+ * Print multiple select field.
719
+ *
720
+ * @param array $arguments Field arguments.
721
+ *
722
+ * @noinspection PhpUnusedPrivateMethodInspection
723
+ */
724
+ private function print_multiple_select_field( array $arguments ) {
725
+ $value = $this->get( $arguments['field_id'] );
726
+
727
+ if ( empty( $arguments['options'] ) || ! is_array( $arguments['options'] ) ) {
728
+ return;
729
+ }
730
+
731
+ $options_markup = '';
732
+ foreach ( $arguments['options'] as $key => $label ) {
733
+ $selected = '';
734
+ if ( is_array( $value ) && in_array( $key, $value, true ) ) {
735
+ $selected = selected( $key, $key, false );
736
+ }
737
+ $options_markup .= sprintf(
738
+ '<option value="%s" %s>%s</option>',
739
+ $key,
740
+ $selected,
741
+ $label
742
+ );
743
+ }
744
+
745
+ printf(
746
+ '<select multiple="multiple" name="%1$s[%2$s][]">%3$s</select>',
747
+ esc_html( $this->option_name() ),
748
+ esc_html( $arguments['field_id'] ),
749
+ wp_kses(
750
+ $options_markup,
751
+ [
752
+ 'option' => [
753
+ 'value' => [],
754
+ 'selected' => [],
755
+ ],
756
+ ]
757
+ )
758
+ );
759
+ }
760
+
761
+ /**
762
+ * Print table field.
763
+ *
764
+ * @param array $arguments Field arguments.
765
+ *
766
+ * @noinspection PhpUnusedPrivateMethodInspection
767
+ */
768
+ private function print_table_field( array $arguments ) {
769
+ $value = $this->get( $arguments['field_id'] );
770
+
771
+ if ( ! is_array( $value ) ) {
772
+ return;
773
+ }
774
+
775
+ $iterator = 0;
776
+ foreach ( $value as $key => $cell_value ) {
777
+ $id = $arguments['field_id'] . '-' . $iterator;
778
+
779
+ echo '<div class="ctl-table-cell">';
780
+ printf(
781
+ '<label for="%1$s">%2$s</label>',
782
+ esc_html( $id ),
783
+ esc_html( $key )
784
+ );
785
+ printf(
786
+ '<input name="%1$s[%2$s][%3$s]" id="%4$s" type="%5$s"' .
787
+ ' placeholder="%6$s" value="%7$s" class="regular-text" />',
788
+ esc_html( $this->option_name() ),
789
+ esc_attr( $arguments['field_id'] ),
790
+ esc_attr( $key ),
791
+ esc_attr( $id ),
792
+ 'text',
793
+ esc_attr( $arguments['placeholder'] ),
794
+ esc_html( $cell_value )
795
+ );
796
+ echo '</div>';
797
+
798
+ $iterator ++;
799
+ }
800
+ }
801
+
802
+ /**
803
+ * Output settings field.
804
+ *
805
+ * @param array $arguments Field arguments.
806
+ */
807
+ public function field_callback( array $arguments ) {
808
+ if ( ! isset( $arguments['field_id'] ) ) {
809
+ return;
810
+ }
811
+
812
+ $types = [
813
+ 'text' => 'print_text_field',
814
+ 'password' => 'print_text_field',
815
+ 'number' => 'print_number_field',
816
+ 'textarea' => 'print_text_area_field',
817
+ 'checkbox' => 'print_check_box_field',
818
+ 'radio' => 'print_radio_field',
819
+ 'select' => 'print_select_field',
820
+ 'multiple' => 'print_multiple_select_field',
821
+ 'table' => 'print_table_field',
822
+ ];
823
+
824
+ $type = $arguments['type'];
825
+
826
+ if ( ! array_key_exists( $type, $types ) ) {
827
+ return;
828
+ }
829
+
830
+ // If there is help text.
831
+ $helper = $arguments['helper'];
832
+ if ( $helper ) {
833
+ printf(
834
+ '<span class="helper"><span class="helper-content">%s</span></span>',
835
+ wp_kses_post( $helper )
836
+ );
837
+ }
838
+
839
+ $this->{$types[ $type ]}( $arguments );
840
+
841
+ // If there is supplemental text.
842
+ $supplemental = $arguments['supplemental'];
843
+ if ( $supplemental ) {
844
+ printf( '<p class="description">%s</p>', wp_kses_post( $supplemental ) );
845
+ }
846
+ }
847
+
848
+ /**
849
+ * Get plugin option.
850
+ *
851
+ * @param string $key Setting name.
852
+ * @param mixed $empty_value Empty value for this setting.
853
+ *
854
+ * @return string|array The value specified for the option or a default value for the option.
855
+ */
856
+ public function get( $key, $empty_value = null ) {
857
+ if ( empty( $this->settings ) ) {
858
+ $this->init_settings();
859
+ }
860
+
861
+ // Get option default if unset.
862
+ if ( ! isset( $this->settings[ $key ] ) ) {
863
+ $form_fields = $this->form_fields();
864
+ $this->settings[ $key ] = isset( $form_fields[ $key ] ) ? $this->field_default( $form_fields[ $key ] ) : '';
865
+ }
866
+
867
+ if ( '' === $this->settings[ $key ] && ! is_null( $empty_value ) ) {
868
+ $this->settings[ $key ] = $empty_value;
869
+ }
870
+
871
+ return $this->settings[ $key ];
872
+ }
873
+
874
+ /**
875
+ * Get a field default value. Defaults to '' if not set.
876
+ *
877
+ * @param array $field Setting field default value.
878
+ *
879
+ * @return string
880
+ */
881
+ protected function field_default( array $field ) {
882
+ return empty( $field['default'] ) ? '' : $field['default'];
883
+ }
884
+
885
+ /**
886
+ * Update plugin option.
887
+ *
888
+ * @param string $key Setting name.
889
+ * @param mixed $value Setting value.
890
+ */
891
+ public function update_option( $key, $value ) {
892
+ if ( empty( $this->settings ) ) {
893
+ $this->init_settings();
894
+ }
895
+
896
+ $this->settings[ $key ] = $value;
897
+ update_option( $this->option_name(), $this->settings );
898
+ }
899
+
900
+ /**
901
+ * Filter plugin option update.
902
+ *
903
+ * @param mixed $value New option value.
904
+ * @param mixed $old_value Old option value.
905
+ *
906
+ * @return mixed
907
+ */
908
+ public function pre_update_option_filter( $value, $old_value ) {
909
+ if ( $value === $old_value ) {
910
+ return $value;
911
+ }
912
+
913
+ // We save only one table, so merge with all existing tables.
914
+ if ( is_array( $old_value ) && ( is_array( $value ) ) ) {
915
+ $value = array_merge( $old_value, $value );
916
+ }
917
+
918
+ $form_fields = $this->form_fields();
919
+ foreach ( $form_fields as $key => $form_field ) {
920
+ if ( 'checkbox' === $form_field['type'] ) {
921
+ $form_field_value = isset( $value[ $key ] ) ? $value[ $key ] : 'no';
922
+ $form_field_value = '1' === $form_field_value || 'yes' === $form_field_value ? 'yes' : 'no';
923
+ $value[ $key ] = $form_field_value;
924
+ }
925
+ }
926
+
927
+ return $value;
928
+ }
929
+
930
+ /**
931
+ * Load plugin text domain.
932
+ */
933
+ public function load_plugin_textdomain() {
934
+ load_plugin_textdomain(
935
+ $this->text_domain(),
936
+ false,
937
+ dirname( $this->plugin_basename() ) . '/languages/'
938
+ );
939
+ }
940
+
941
+ /**
942
+ * Is current admin screen the plugin options screen.
943
+ *
944
+ * @return bool
945
+ */
946
+ protected function is_options_screen() {
947
+ $current_screen = get_current_screen();
948
+
949
+ $screen_id = $this->screen_id();
950
+ if ( $this->is_main_menu_page() ) {
951
+ $screen_id = str_replace( 'settings_page', 'toplevel_page', $screen_id );
952
+ }
953
+
954
+ return $current_screen && ( 'options' === $current_screen->id || $screen_id === $current_screen->id );
955
+ }
956
+ }
src/php/Settings/Abstracts/SettingsInterface.php ADDED
@@ -0,0 +1,23 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ /**
3
+ * SettingsInterface interface file.
4
+ *
5
+ * @package cyr-to-lat
6
+ */
7
+
8
+ namespace Cyr_To_Lat\Settings\Abstracts;
9
+
10
+ /**
11
+ * Interface SettingsInterface.
12
+ */
13
+ interface SettingsInterface {
14
+ /**
15
+ * Get plugin option.
16
+ *
17
+ * @param string $key Setting name.
18
+ * @param mixed $empty_value Empty value for this setting.
19
+ *
20
+ * @return string|array The value specified for the option or a default value for the option.
21
+ */
22
+ public function get( $key, $empty_value = null );
23
+ }
src/php/Settings/Converter.php ADDED
@@ -0,0 +1,295 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ /**
3
+ * Converter class file.
4
+ *
5
+ * @package cyr-to-lat
6
+ */
7
+
8
+ namespace Cyr_To_Lat\Settings;
9
+
10
+ /**
11
+ * Class Converter
12
+ *
13
+ * Settings page "Converter".
14
+ */
15
+ class Converter extends PluginSettingsBase {
16
+
17
+ /**
18
+ * Admin script handle.
19
+ */
20
+ const HANDLE = 'cyr-to-lat-settings';
21
+
22
+ /**
23
+ * Converter nonce.
24
+ */
25
+ const NONCE = 'cyr-to-lat-converter-nonce';
26
+
27
+ /**
28
+ * Get screen id.
29
+ *
30
+ * @return string
31
+ */
32
+ public function screen_id() {
33
+ return 'settings_page_cyr-to-lat';
34
+ }
35
+
36
+ /**
37
+ * Get option group.
38
+ *
39
+ * @return string
40
+ */
41
+ protected function option_group() {
42
+ return 'cyr_to_lat_group';
43
+ }
44
+
45
+ /**
46
+ * Get option page.
47
+ *
48
+ * @return string
49
+ */
50
+ protected function option_page() {
51
+ return 'cyr-to-lat';
52
+ }
53
+
54
+ /**
55
+ * Get option name.
56
+ *
57
+ * @return string
58
+ */
59
+ protected function option_name() {
60
+ return 'cyr_to_lat_settings';
61
+ }
62
+
63
+ /**
64
+ * Get page title.
65
+ *
66
+ * @return string
67
+ */
68
+ protected function page_title() {
69
+ return __( 'Converter', 'cyr2lat' );
70
+ }
71
+
72
+ /**
73
+ * Get menu title.
74
+ *
75
+ * @return string
76
+ */
77
+ protected function menu_title() {
78
+ return __( 'Cyr To Lat', 'cyr2lat' );
79
+ }
80
+
81
+ /**
82
+ * Get section title.
83
+ *
84
+ * @return string
85
+ */
86
+ protected function section_title() {
87
+ return '';
88
+ }
89
+
90
+ /**
91
+ * Get parent slug.
92
+ *
93
+ * @return string
94
+ */
95
+ protected function parent_slug() {
96
+ return 'options-general.php';
97
+ }
98
+
99
+ /**
100
+ * Init form fields.
101
+ */
102
+ public function init_form_fields() {
103
+ $default_post_types = [ 'post', 'page', 'nav_menu_item' ];
104
+ $post_types = get_post_types( [ 'public' => true ] );
105
+
106
+ $post_types += [ 'nav_menu_item' => 'nav_menu_item' ];
107
+
108
+ $filtered_post_types = apply_filters( 'ctl_post_types', $post_types );
109
+
110
+ $this->form_fields = [];
111
+
112
+ $this->form_fields['background_post_types'] = [
113
+ 'label' => __( 'Post Types', 'cyr2lat' ),
114
+ 'section' => 'background_section',
115
+ 'type' => 'checkbox',
116
+ 'placeholder' => '',
117
+ 'helper' => __( 'Post types included in the conversion.', 'cyr2lat' ),
118
+ 'supplemental' => '',
119
+ 'options' => [],
120
+ ];
121
+
122
+ foreach ( $post_types as $post_type ) {
123
+ $label = $post_type;
124
+
125
+ $this->form_fields['background_post_types']['options'][ $post_type ] = $label;
126
+ }
127
+
128
+ $this->form_fields['background_post_types']['default'] = $default_post_types;
129
+ $this->form_fields['background_post_types']['disabled'] = array_diff( $default_post_types, $filtered_post_types );
130
+
131
+ $default_post_statuses = [ 'publish', 'future', 'private' ];
132
+ $post_statuses = [ 'publish', 'future', 'private', 'draft', 'pending' ];
133
+
134
+ $this->form_fields['background_post_statuses'] = [
135
+ 'label' => __( 'Post Statuses', 'cyr2lat' ),
136
+ 'section' => 'background_section',
137
+ 'type' => 'checkbox',
138
+ 'placeholder' => '',
139
+ 'helper' => __( 'Post statuses included in the conversion.', 'cyr2lat' ),
140
+ 'supplemental' => '',
141
+ 'options' => [],
142
+ ];
143
+
144
+ foreach ( $post_statuses as $post_status ) {
145
+ $label = $post_status;
146
+
147
+ $this->form_fields['background_post_statuses']['options'][ $post_status ] = $label;
148
+ }
149
+
150
+ $this->form_fields['background_post_statuses']['default'] = $default_post_statuses;
151
+ }
152
+
153
+ /**
154
+ * Init class hooks.
155
+ */
156
+ protected function init_hooks() {
157
+ parent::init_hooks();
158
+
159
+ add_action( 'in_admin_header', [ $this, 'in_admin_header' ] );
160
+ }
161
+
162
+ /**
163
+ * Show settings page.
164
+ */
165
+ public function settings_page() {
166
+ ?>
167
+ <div class="wrap">
168
+ <h1>
169
+ <?php
170
+ esc_html_e( 'Cyr To Lat Plugin Options', 'cyr2lat' );
171
+ ?>
172
+ </h1>
173
+
174
+ <form id="ctl-options" action="<?php echo esc_url( admin_url( 'options.php' ) ); ?>" method="post">
175
+ <?php
176
+ do_settings_sections( $this->option_page() ); // Sections with options.
177
+ settings_fields( $this->option_group() ); // Hidden protection fields.
178
+ submit_button();
179
+ ?>
180
+ </form>
181
+
182
+ <form id="ctl-convert-existing-slugs" action="" method="post">
183
+ <input type="hidden" name="ctl-convert" />
184
+ <?php
185
+ wp_nonce_field( self::NONCE );
186
+ submit_button( __( 'Convert Existing Slugs', 'cyr2lat' ), 'secondary', 'ctl-convert-button' );
187
+ ?>
188
+ </form>
189
+ </div>
190
+ <?php
191
+ }
192
+
193
+ /**
194
+ * Section callback.
195
+ *
196
+ * @param array $arguments Section arguments.
197
+ */
198
+ public function section_callback( $arguments ) {
199
+ if ( 'background_section' === $arguments['id'] ) {
200
+ ?>
201
+ <h2 class="title">
202
+ <?php
203
+ esc_html_e( 'Existing Slugs Conversion Settings', 'cyr2lat' );
204
+ ?>
205
+ </h2>
206
+ <p>
207
+ <?php
208
+ echo wp_kses_post(
209
+ __(
210
+ 'Existing <strong>product attribute</strong> slugs will <strong>NOT</strong> be converted.',
211
+ 'cyr2lat'
212
+ )
213
+ );
214
+ ?>
215
+ </p>
216
+ <?php
217
+ }
218
+ }
219
+
220
+ /**
221
+ * Output convert confirmation popup.
222
+ */
223
+ public function in_admin_header() {
224
+ if ( ! $this->is_options_screen() ) {
225
+ return;
226
+ }
227
+
228
+ ?>
229
+ <div id="ctl-confirm-popup">
230
+ <div id="ctl-confirm-content">
231
+ <p>
232
+ <strong><?php esc_html_e( 'Important:', 'cyr2lat' ); ?></strong>
233
+ <?php
234
+ esc_html_e(
235
+ 'This operation is irreversible. Please make sure that you have made a backup copy of your database.',
236
+ 'cyr2lat'
237
+ );
238
+ ?>
239
+ </p>
240
+ <p>
241
+ <?php
242
+ esc_html_e(
243
+ 'Also, you have to make a copy of your media files if the attachment post type is selected for
244
+ conversion.',
245
+ 'cyr2lat'
246
+ );
247
+ ?>
248
+ </p>
249
+ <p>
250
+ <?php
251
+ esc_html_e(
252
+ 'Upon conversion of attachments, please regenerate thumbnails.',
253
+ 'cyr2lat'
254
+ );
255
+ ?>
256
+ </p>
257
+ <p><?php esc_html_e( 'Are you sure to continue?', 'cyr2lat' ); ?></p>
258
+ <div id="ctl-confirm-buttons">
259
+ <input
260
+ type="button" id="ctl-confirm-ok" class="button button-primary"
261
+ value="<?php esc_html_e( 'OK', 'cyr2lat' ); ?>">
262
+ <button
263
+ type="button" id="ctl-confirm-cancel" class="button button-secondary">
264
+ <?php esc_html_e( 'Cancel', 'cyr2lat' ); ?>
265
+ </button>
266
+ </div>
267
+ </div>
268
+ </div>
269
+ <?php
270
+ }
271
+
272
+ /**
273
+ * Enqueue class scripts.
274
+ */
275
+ public function admin_enqueue_scripts() {
276
+ if ( ! $this->is_options_screen() ) {
277
+ return;
278
+ }
279
+
280
+ wp_enqueue_script(
281
+ self::HANDLE,
282
+ constant( 'CYR_TO_LAT_URL' ) . '/assets/js/converter/app.js',
283
+ [],
284
+ constant( 'CYR_TO_LAT_VERSION' ),
285
+ true
286
+ );
287
+
288
+ wp_enqueue_style(
289
+ self::HANDLE,
290
+ constant( 'CYR_TO_LAT_URL' ) . '/assets/css/converter.css',
291
+ [],
292
+ constant( 'CYR_TO_LAT_VERSION' )
293
+ );
294
+ }
295
+ }
src/php/Settings/PluginSettingsBase.php ADDED
@@ -0,0 +1,72 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ /**
3
+ * PluginSettingsBase class file.
4
+ *
5
+ * @package cyr-to-lat
6
+ */
7
+
8
+ namespace Cyr_To_Lat\Settings;
9
+
10
+ use Cyr_To_Lat\Settings\Abstracts\SettingsBase;
11
+
12
+ /**
13
+ * Class PluginSettingsBase
14
+ *
15
+ * Extends general SettingsBase suitable for any plugin with current plugin related methods.
16
+ */
17
+ abstract class PluginSettingsBase extends SettingsBase {
18
+
19
+ /**
20
+ * Get plugin base name.
21
+ *
22
+ * @return string
23
+ */
24
+ protected function plugin_basename() {
25
+ return plugin_basename( constant( 'CYR_TO_LAT_FILE' ) );
26
+ }
27
+
28
+ /**
29
+ * Get plugin url.
30
+ *
31
+ * @return string
32
+ */
33
+ protected function plugin_url() {
34
+ return constant( 'CYR_TO_LAT_URL' );
35
+ }
36
+
37
+ /**
38
+ * Get plugin version.
39
+ *
40
+ * @return string
41
+ */
42
+ protected function plugin_version() {
43
+ return constant( 'CYR_TO_LAT_VERSION' );
44
+ }
45
+
46
+ /**
47
+ * Get settings link label.
48
+ *
49
+ * @return string
50
+ */
51
+ protected function settings_link_label() {
52
+ return __( 'View Cyr To Lat settings', 'cyr2lat' );
53
+ }
54
+
55
+ /**
56
+ * Get settings link text.
57
+ *
58
+ * @return string
59
+ */
60
+ protected function settings_link_text() {
61
+ return __( 'Settings', 'cyr2lat' );
62
+ }
63
+
64
+ /**
65
+ * Get text domain.
66
+ *
67
+ * @return string
68
+ */
69
+ protected function text_domain() {
70
+ return 'cyr2lat';
71
+ }
72
+ }
src/php/Settings/Settings.php ADDED
@@ -0,0 +1,165 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ /**
3
+ * Settings class file.
4
+ *
5
+ * @package cyr-to-lat
6
+ */
7
+
8
+ namespace Cyr_To_Lat\Settings;
9
+
10
+ use Cyr_To_Lat\Settings\Abstracts\SettingsBase;
11
+ use Cyr_To_Lat\Settings\Abstracts\SettingsInterface;
12
+ use Cyr_To_Lat\Symfony\Polyfill\Mbstring\Mbstring;
13
+
14
+ /**
15
+ * Class Settings
16
+ *
17
+ * Central point to get settings from.
18
+ */
19
+ class Settings implements SettingsInterface {
20
+
21
+ /**
22
+ * Menu pages classes.
23
+ */
24
+ const MENU_PAGES = [
25
+ [ Tables::class, Converter::class ],
26
+ ];
27
+
28
+ /**
29
+ * Menu pages class instances.
30
+ *
31
+ * @var array
32
+ */
33
+ protected $menu_pages = [];
34
+
35
+ /**
36
+ * Screen ids of pages and tabs.
37
+ *
38
+ * @var array
39
+ */
40
+ private $screen_ids = [];
41
+
42
+ /**
43
+ * Settings constructor.
44
+ */
45
+ public function __construct() {
46
+ $this->init();
47
+ }
48
+
49
+ /**
50
+ * Init class.
51
+ */
52
+ protected function init() {
53
+ // Allow to specify MENU_PAGES item as one class, not an array.
54
+ $menu_pages = (array) self::MENU_PAGES;
55
+
56
+ foreach ( $menu_pages as $menu_page ) {
57
+ $tab_classes = (array) $menu_page;
58
+
59
+ // Allow to specify menu page as one class, without tabs.
60
+ $page_class = $tab_classes[0];
61
+ $tab_classes = array_slice( $tab_classes, 1 );
62
+
63
+ $tabs = [];
64
+ foreach ( $tab_classes as $tab_class ) {
65
+ /**
66
+ * Tab.
67
+ *
68
+ * @var PluginSettingsBase $tab
69
+ */
70
+ $tab = new $tab_class( null );
71
+ $tabs[] = $tab;
72
+ $this->screen_ids[] = $tab->screen_id();
73
+ }
74
+
75
+ /**
76
+ * Page.
77
+ *
78
+ * @var PluginSettingsBase $page_class
79
+ */
80
+ $this->menu_pages[] = new $page_class( $tabs );
81
+ }
82
+ }
83
+
84
+ /**
85
+ * Get plugin option.
86
+ *
87
+ * @param string $key Setting name.
88
+ * @param mixed $empty_value Empty value for this setting.
89
+ *
90
+ * @return string|array The value specified for the option or a default value for the option.
91
+ */
92
+ public function get( $key, $empty_value = null ) {
93
+ return $this->get_main_page()->get( $key, $empty_value );
94
+ }
95
+
96
+ /**
97
+ * Get main admin page.
98
+ *
99
+ * @return SettingsBase
100
+ */
101
+ private function get_main_page() {
102
+ return $this->menu_pages[0];
103
+ }
104
+
105
+ /**
106
+ * Get screen ids of all settings pages and tabs.
107
+ *
108
+ * @return array
109
+ */
110
+ public function screen_ids() {
111
+ return $this->screen_ids;
112
+ }
113
+
114
+ /**
115
+ * Get transliteration table.
116
+ *
117
+ * @return array
118
+ */
119
+ public function get_table() {
120
+ // List of locales: https://make.wordpress.org/polyglots/teams/.
121
+ $locale = get_locale();
122
+ $table = $this->get( $locale );
123
+ if ( empty( $table ) ) {
124
+ $table = $this->get( 'iso9' );
125
+ }
126
+
127
+ return $this->transpose_chinese_table( $table );
128
+ }
129
+
130
+ /**
131
+ * Is current locale a Chinese one.
132
+ *
133
+ * @return bool
134
+ */
135
+ public function is_chinese_locale() {
136
+ $chinese_locales = [ 'zh_CN', 'zh_HK', 'zh_SG', 'zh_TW' ];
137
+
138
+ return in_array( get_locale(), $chinese_locales, true );
139
+ }
140
+
141
+ /**
142
+ * Transpose Chinese table.
143
+ *
144
+ * Chinese tables are stored in different way, to show them compact.
145
+ *
146
+ * @param array $table Table.
147
+ *
148
+ * @return array
149
+ */
150
+ protected function transpose_chinese_table( $table ) {
151
+ if ( ! $this->is_chinese_locale() ) {
152
+ return $table;
153
+ }
154
+
155
+ $transposed_table = [];
156
+ foreach ( $table as $key => $item ) {
157
+ $hieroglyphs = Mbstring::mb_str_split( $item );
158
+ foreach ( $hieroglyphs as $hieroglyph ) {
159
+ $transposed_table[ $hieroglyph ] = $key;
160
+ }
161
+ }
162
+
163
+ return $transposed_table;
164
+ }
165
+ }
src/php/Settings/Tables.php ADDED
@@ -0,0 +1,286 @@
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
+ <?php
2
+ /**
3
+ * Tables class file.
4
+ *
5
+ * @package cyr-to-lat
6
+ */
7
+
8
+ namespace Cyr_To_Lat\Settings;
9
+
10
+ use Cyr_To_Lat\Conversion_Tables;
11
+
12
+ /**
13
+ * Class Tables
14
+ *
15
+ * Settings page "Tables" (main).
16
+ */
17
+ class Tables extends PluginSettingsBase {
18
+
19
+ /**
20
+ * Admin script handle.
21
+ */
22
+ const HANDLE = 'cyr-to-lat-tables';
23
+
24
+ /**
25
+ * Script localization object.
26
+ */
27
+ const OBJECT = ' Cyr2LatTablesObject';
28
+
29
+ /**
30
+ * Served locales.
31
+ *
32
+ * @var array
33
+ */
34
+ protected $locales = [];
35
+
36
+ /**
37
+ * Get screen id.
38
+ *
39
+ * @return string
40
+ */
41
+ public function screen_id() {
42
+ return 'settings_page_cyr-to-lat';
43
+ }
44
+
45
+ /**
46
+ * Get option group.
47
+ *
48
+ * @return string
49
+ */
50
+ protected function option_group() {
51
+ return 'cyr_to_lat_group';
52
+ }
53
+
54
+ /**
55
+ * Get option page.
56
+ *
57
+ * @return string
58
+ */
59
+ protected function option_page() {
60
+ return 'cyr-to-lat';
61
+ }
62
+
63
+ /**
64
+ * Get option name.
65
+ *
66
+ * @return string
67
+ */
68
+ protected function option_name() {
69
+ return 'cyr_to_lat_settings';
70
+ }
71
+
72
+ /**
73
+ * Get page title.
74
+ *
75
+ * @return string
76
+ */
77
+ protected function page_title() {
78
+ return __( 'Tables', 'cyr2lat' );
79
+ }
80
+
81
+ /**
82
+ * Get menu title.
83
+ *
84
+ * @return string
85
+ */
86
+ protected function menu_title() {
87
+ return __( 'Cyr To Lat', 'cyr2lat' );
88
+ }
89
+
90
+ /**
91
+ * Get section title.
92
+ *
93
+ * @return string
94
+ */
95
+ protected function section_title() {
96
+ return '';
97
+ }
98
+
99
+ /**
100
+ * Get parent slug.
101
+ *
102
+ * @return string
103
+ */
104
+ protected function parent_slug() {
105
+ // Indicate that it is the main menu page.
106
+ return 'options-general.php';
107
+ }
108
+
109
+ /**
110
+ * Init locales.
111
+ */
112
+ protected function init_locales() {
113
+ if ( ! empty( $this->locales ) ) {
114
+ return;
115
+ }
116
+
117
+ $this->locales = [
118
+ 'iso9' => [
119
+ 'label' => __( 'ISO9 Table', 'cyr2lat' ),
120
+ ],
121
+ 'bel' => [
122
+ 'label' => __( 'bel Table', 'cyr2lat' ),
123
+ ],
124
+ 'uk' => [
125
+ 'label' => __( 'uk Table', 'cyr2lat' ),
126
+ ],
127
+ 'bg_BG' => [
128
+ 'label' => __( 'bg_BG Table', 'cyr2lat' ),
129
+ ],
130
+ 'mk_MK' => [
131
+ 'label' => __( 'mk_MK Table', 'cyr2lat' ),
132
+ ],
133
+ 'sr_RS' => [
134
+ 'label' => __( 'sr_RS Table', 'cyr2lat' ),
135
+ ],
136
+ 'el' => [
137
+ 'label' => __( 'el Table', 'cyr2lat' ),
138
+ ],
139
+ 'hy' => [
140
+ 'label' => __( 'hy Table', 'cyr2lat' ),
141
+ ],
142
+ 'ka_GE' => [
143
+ 'label' => __( 'ka_GE Table', 'cyr2lat' ),
144
+ ],
145
+ 'kk' => [
146
+ 'label' => __( 'kk Table', 'cyr2lat' ),
147
+ ],
148
+ 'he_IL' => [
149
+ 'label' => __( 'he_IL Table', 'cyr2lat' ),
150
+ ],
151
+ 'zh_CN' => [
152
+ 'label' => __( 'zh_CN Table', 'cyr2lat' ),
153
+ ],
154
+ ];
155
+ }
156
+
157
+ /**
158
+ * Get current locale.
159
+ *
160
+ * @return string
161
+ */
162
+ private function get_current_locale() {
163
+ $current_locale = get_locale();
164
+
165
+ return array_key_exists( $current_locale, $this->locales ) ? $current_locale : 'iso9';
166
+ }
167
+
168
+ /**
169
+ * Init form fields.
170
+ */
171
+ public function init_form_fields() {
172
+ $this->init_locales();
173
+
174
+ $current_locale = $this->get_current_locale();
175
+
176
+ $this->form_fields = [];
177
+
178
+ foreach ( $this->locales as $locale => $info ) {
179
+ $current = ( $locale === $current_locale ) ? '<br>' . __( '(current)', 'cyr2lat' ) : '';
180
+
181
+ $this->form_fields[ $locale ] = [
182
+ 'label' => $info['label'] . $current,
183
+ 'section' => $locale . '_section',
184
+ 'type' => 'table',
185
+ 'placeholder' => '',
186
+ 'helper' => '',
187
+ 'supplemental' => '',
188
+ 'default' => Conversion_Tables::get( $locale ),
189
+ ];
190
+ }
191
+ }
192
+
193
+ /**
194
+ * Show settings page.
195
+ */
196
+ public function settings_page() {
197
+ ?>
198
+ <div class="wrap">
199
+ <h1>
200
+ <?php
201
+ // Admin panel title.
202
+ esc_html_e( 'Cyr To Lat Plugin Options', 'cyr2lat' );
203
+ ?>
204
+ </h1>
205
+
206
+ <form id="ctl-options" action="<?php echo esc_url( admin_url( 'options.php' ) ); ?>" method="post">
207
+ <?php
208
+ do_settings_sections( $this->option_page() ); // Sections with options.
209
+ settings_fields( $this->option_group() ); // Hidden protection fields.
210
+ submit_button();
211
+ ?>
212
+ </form>
213
+
214
+ <div id="appreciation">
215
+ <h2>
216
+ <?php echo esc_html( __( 'Your Appreciation', 'cyr2lat' ) ); ?>
217
+ </h2>
218
+ <a
219
+ target="_blank"
220
+ href="https://wordpress.org/support/view/plugin-reviews/cyr2lat?rate=5#postform">
221
+ <?php echo esc_html( __( 'Leave a ★★★★★ plugin review on WordPress.org', 'cyr2lat' ) ); ?>
222
+ </a>
223
+ </div>
224
+ </div>
225
+ <?php
226
+ }
227
+
228
+ /**
229
+ * Section callback.
230
+ *
231
+ * @param array $arguments Section arguments.
232
+ */
233
+ public function section_callback( $arguments ) {
234
+ $locale = str_replace( '_section', '', $arguments['id'] );
235
+ if ( $this->get_current_locale() === $locale ) {
236
+ echo '<div id="ctl-current"></div>';
237
+ }
238
+ }
239
+
240
+ /**
241
+ * Enqueue class scripts.
242
+ */
243
+ public function admin_enqueue_scripts() {
244
+ if ( ! $this->is_options_screen() ) {
245
+ return;
246
+ }
247
+
248
+ wp_enqueue_script(
249
+ self::HANDLE,
250
+ constant( 'CYR_TO_LAT_URL' ) . '/assets/js/tables/app.js',
251
+ [],
252
+ constant( 'CYR_TO_LAT_VERSION' ),
253
+ true
254
+ );
255
+
256
+ wp_localize_script(
257
+ self::HANDLE,
258
+ self::OBJECT,
259
+ [
260
+ 'optionsSaveSuccessMessage' => __( 'Options saved.', 'cyr2lat' ),
261
+ 'optionsSaveErrorMessage' => __( 'Error saving options.', 'cyr2lat' ),
262
+ ]
263
+ );
264
+
265
+ wp_enqueue_style(
266
+ self::HANDLE,
267
+ constant( 'CYR_TO_LAT_URL' ) . '/assets/css/tables.css',
268
+ [],
269
+ constant( 'CYR_TO_LAT_VERSION' )
270
+ );
271
+ }
272
+
273
+ /**
274
+ * Setup settings sections.
275
+ */
276
+ public function setup_sections() {
277
+ foreach ( $this->form_fields as $form_field ) {
278
+ add_settings_section(
279
+ $form_field['section'],
280
+ $form_field['label'],
281
+ [ $this, 'section_callback' ],
282
+ $this->option_page()
283
+ );
284
+ }
285
+ }
286
+ }
{classes → src/php}/background-processes/class-conversion-process.php RENAMED
File without changes
{classes → src/php}/background-processes/class-post-conversion-process.php RENAMED
File without changes
{classes → src/php}/background-processes/class-term-conversion-process.php RENAMED
File without changes
{classes → src/php}/class-acf.php RENAMED
@@ -7,6 +7,8 @@
7
 
8
  namespace Cyr_To_Lat;
9
 
 
 
10
  /**
11
  * Class ACF
12
  */
@@ -45,7 +47,7 @@ class ACF {
45
 
46
  wp_enqueue_script(
47
  'cyr-to-lat-acf-field-group',
48
- constant( 'CYR_TO_LAT_URL' ) . '/js/acf-field-group.js',
49
  [],
50
  constant( 'CYR_TO_LAT_VERSION' ),
51
  true
7
 
8
  namespace Cyr_To_Lat;
9
 
10
+ use Cyr_To_Lat\Settings\Settings;
11
+
12
  /**
13
  * Class ACF
14
  */
47
 
48
  wp_enqueue_script(
49
  'cyr-to-lat-acf-field-group',
50
+ constant( 'CYR_TO_LAT_URL' ) . '/assets/js/acf-field-group.js',
51
  [],
52
  constant( 'CYR_TO_LAT_VERSION' ),
53
  true
{classes → src/php}/class-admin-notices.php RENAMED
@@ -49,7 +49,7 @@ class Admin_Notices {
49
  */
50
  public function show_notices() {
51
  foreach ( $this->notices as $notice ) {
52
- if ( ! $this->is_page_allowed( $notice ) ) {
53
  continue;
54
  }
55
 
@@ -72,17 +72,16 @@ class Admin_Notices {
72
  *
73
  * @return bool
74
  */
75
- protected function is_page_allowed( $notice ) {
76
- $page = isset( $notice['options']['page'] ) ? $notice['options']['page'] : null;
77
- if ( ! $page ) {
78
  return true;
79
  }
80
- $pages = (array) $page;
81
 
82
  $current_screen = get_current_screen();
83
 
84
- foreach ( $pages as $page ) {
85
- if ( $current_screen && $page === $current_screen->id ) {
86
  return true;
87
  }
88
  }
49
  */
50
  public function show_notices() {
51
  foreach ( $this->notices as $notice ) {
52
+ if ( ! $this->is_screen_allowed( $notice ) ) {
53
  continue;
54
  }
55
 
72
  *
73
  * @return bool
74
  */
75
+ protected function is_screen_allowed( $notice ) {
76
+ $screen_ids = isset( $notice['options']['screen_ids'] ) ? (array) $notice['options']['screen_ids'] : null;
77
+ if ( empty( $screen_ids ) ) {
78
  return true;
79
  }
 
80
 
81
  $current_screen = get_current_screen();
82
 
83
+ foreach ( $screen_ids as $screen_id ) {
84
+ if ( $current_screen && $screen_id === $current_screen->id ) {
85
  return true;
86
  }
87
  }
{classes → src/php}/class-conversion-tables.php RENAMED
File without changes
{classes → src/php}/class-converter.php RENAMED
@@ -7,6 +7,8 @@
7
 
8
  namespace Cyr_To_Lat;
9
 
 
 
10
  /**
11
  * Class Converter
12
  *
@@ -34,47 +36,48 @@ class Converter {
34
  *
35
  * @var Main
36
  */
37
- private $main;
 
 
 
 
 
 
 
38
 
39
  /**
40
  * Background process to convert posts.
41
  *
42
  * @var Post_Conversion_Process
43
  */
44
- private $process_all_posts;
45
 
46
  /**
47
  * Background process to convert terms.
48
  *
49
  * @var Term_Conversion_Process
50
  */
51
- private $process_all_terms;
52
 
53
  /**
54
  * Admin notices.
55
  *
56
  * @var Admin_Notices
57
  */
58
- private $admin_notices;
59
-
60
- /**
61
- * Option group.
62
- *
63
- * @var string
64
- */
65
- private $option_group;
66
 
67
  /**
68
  * Converter constructor.
69
  *
70
  * @param Main $main Plugin main class.
 
71
  * @param Post_Conversion_Process $process_all_posts Post conversion process.
72
  * @param Term_Conversion_Process $process_all_terms Term conversion process.
73
  * @param Admin_Notices $admin_notices Admin notices.
74
  */
75
- public function __construct( $main, $process_all_posts, $process_all_terms, $admin_notices ) {
76
  $this->main = $main;
77
- $this->option_group = Settings::OPTION_GROUP;
78
  $this->process_all_posts = $process_all_posts;
79
  $this->process_all_terms = $process_all_terms;
80
  $this->admin_notices = $admin_notices;
@@ -137,7 +140,7 @@ class Converter {
137
  if ( ! isset( $_POST['ctl-convert'] ) ) {
138
  return;
139
  }
140
- check_admin_referer( $this->option_group . '-options' );
141
  $this->convert_existing_slugs();
142
  }
143
 
@@ -174,13 +177,12 @@ class Converter {
174
  protected function convert_existing_post_slugs( $args = [] ) {
175
  global $wpdb;
176
 
177
- $post_types = get_post_types( [ 'public' => true ] );
178
-
179
- $post_types += [ 'nav_menu_item' => 'nav_menu_item' ];
180
 
181
  $defaults = [
182
  'post_type' => apply_filters( 'ctl_post_types', $post_types ),
183
- 'post_status' => [ 'publish', 'future', 'private' ],
184
  ];
185
 
186
  $parsed_args = wp_parse_args( $args, $defaults );
@@ -191,7 +193,7 @@ class Converter {
191
  'post_status IN (' . $this->main->prepare_in( $parsed_args['post_status'] ) . ')' .
192
  ' AND post_type IN (' . $this->main->prepare_in( $parsed_args['post_type'] ) . ')';
193
 
194
- if ( isset( $parsed_args['post_type']['attachment'] ) ) {
195
  $media_sql = "post_status = 'inherit' AND post_type = 'attachment'";
196
  $post_sql = '(' . $post_sql . ') OR (' . $media_sql . ')';
197
  }
7
 
8
  namespace Cyr_To_Lat;
9
 
10
+ use Cyr_To_Lat\Settings\Settings;
11
+
12
  /**
13
  * Class Converter
14
  *
36
  *
37
  * @var Main
38
  */
39
+ protected $main;
40
+
41
+ /**
42
+ * Plugin settings class.
43
+ *
44
+ * @var Settings
45
+ */
46
+ protected $settings;
47
 
48
  /**
49
  * Background process to convert posts.
50
  *
51
  * @var Post_Conversion_Process
52
  */
53
+ protected $process_all_posts;
54
 
55
  /**
56
  * Background process to convert terms.
57
  *
58
  * @var Term_Conversion_Process
59
  */
60
+ protected $process_all_terms;
61
 
62
  /**
63
  * Admin notices.
64
  *
65
  * @var Admin_Notices
66
  */
67
+ protected $admin_notices;
 
 
 
 
 
 
 
68
 
69
  /**
70
  * Converter constructor.
71
  *
72
  * @param Main $main Plugin main class.
73
+ * @param Settings $settings Plugin settings.
74
  * @param Post_Conversion_Process $process_all_posts Post conversion process.
75
  * @param Term_Conversion_Process $process_all_terms Term conversion process.
76
  * @param Admin_Notices $admin_notices Admin notices.
77
  */
78
+ public function __construct( $main, $settings, $process_all_posts, $process_all_terms, $admin_notices ) {
79
  $this->main = $main;
80
+ $this->settings = $settings;
81
  $this->process_all_posts = $process_all_posts;
82
  $this->process_all_terms = $process_all_terms;
83
  $this->admin_notices = $admin_notices;
140
  if ( ! isset( $_POST['ctl-convert'] ) ) {
141
  return;
142
  }
143
+ check_admin_referer( \Cyr_To_Lat\Settings\Converter::NONCE );
144
  $this->convert_existing_slugs();
145
  }
146
 
177
  protected function convert_existing_post_slugs( $args = [] ) {
178
  global $wpdb;
179
 
180
+ $post_types = $this->settings->get( 'background_post_types' );
181
+ $post_statuses = $this->settings->get( 'background_post_statuses' );
 
182
 
183
  $defaults = [
184
  'post_type' => apply_filters( 'ctl_post_types', $post_types ),
185
+ 'post_status' => $post_statuses,
186
  ];
187
 
188
  $parsed_args = wp_parse_args( $args, $defaults );
193
  'post_status IN (' . $this->main->prepare_in( $parsed_args['post_status'] ) . ')' .
194
  ' AND post_type IN (' . $this->main->prepare_in( $parsed_args['post_type'] ) . ')';
195
 
196
+ if ( in_array( 'attachment', $parsed_args['post_type'], true ) ) {
197
  $media_sql = "post_status = 'inherit' AND post_type = 'attachment'";
198
  $post_sql = '(' . $post_sql . ') OR (' . $media_sql . ')';
199
  }
{classes → src/php}/class-main.php RENAMED
@@ -9,6 +9,7 @@ namespace Cyr_To_Lat;
9
 
10
  use wpdb;
11
  use Exception;
 
12
  use Cyr_To_Lat\Symfony\Polyfill\Mbstring\Mbstring;
13
 
14
  /**
@@ -76,12 +77,19 @@ class Main {
76
  * Main constructor.
77
  */
78
  public function __construct() {
79
- $this->settings = new Settings();
 
 
 
 
 
 
 
80
  $this->process_all_posts = new Post_Conversion_Process( $this );
81
  $this->process_all_terms = new Term_Conversion_Process( $this );
82
- $this->admin_notices = new Admin_Notices();
83
  $this->converter = new Converter(
84
  $this,
 
85
  $this->process_all_posts,
86
  $this->process_all_terms,
87
  $this->admin_notices
@@ -92,8 +100,6 @@ class Main {
92
  }
93
 
94
  $this->acf = new ACF( $this->settings );
95
-
96
- $this->init();
97
  }
98
 
99
  /**
@@ -468,17 +474,20 @@ class Main {
468
  // phpcs:disable WordPress.Security.NonceVerification.Recommended
469
  if ( isset( $_POST['post_ID'] ) ) {
470
  $pll_get_post_language = pll_get_post_language(
471
- (int) filter_input( INPUT_POST, 'post_ID', FILTER_SANITIZE_STRING )
 
472
  );
473
  }
474
  if ( isset( $_POST['pll_post_id'] ) ) {
475
  $pll_get_post_language = pll_get_post_language(
476
- (int) filter_input( INPUT_POST, 'pll_post_id', FILTER_SANITIZE_STRING )
 
477
  );
478
  }
479
  if ( isset( $_GET['post'] ) ) {
480
  $pll_get_post_language = pll_get_post_language(
481
- (int) filter_input( INPUT_GET, 'post', FILTER_SANITIZE_STRING )
 
482
  );
483
  }
484
  // phpcs:enable WordPress.Security.NonceVerification.Recommended
@@ -506,7 +515,7 @@ class Main {
506
  );
507
 
508
  if ( $pll_get_language ) {
509
- $pll_get_term_language = $pll_get_language->slug;
510
  }
511
  }
512
  // phpcs:enable WordPress.Security.NonceVerification.Missing
9
 
10
  use wpdb;
11
  use Exception;
12
+ use Cyr_To_Lat\Settings\Settings;
13
  use Cyr_To_Lat\Symfony\Polyfill\Mbstring\Mbstring;
14
 
15
  /**
77
  * Main constructor.
78
  */
79
  public function __construct() {
80
+ $this->settings = new Settings();
81
+ $this->admin_notices = new Admin_Notices();
82
+ $requirements = new Requirements( $this->settings, $this->admin_notices );
83
+
84
+ if ( ! $requirements->are_requirements_met() ) {
85
+ return;
86
+ }
87
+
88
  $this->process_all_posts = new Post_Conversion_Process( $this );
89
  $this->process_all_terms = new Term_Conversion_Process( $this );
 
90
  $this->converter = new Converter(
91
  $this,
92
+ $this->settings,
93
  $this->process_all_posts,
94
  $this->process_all_terms,
95
  $this->admin_notices
100
  }
101
 
102
  $this->acf = new ACF( $this->settings );
 
 
103
  }
104
 
105
  /**
474
  // phpcs:disable WordPress.Security.NonceVerification.Recommended
475
  if ( isset( $_POST['post_ID'] ) ) {
476
  $pll_get_post_language = pll_get_post_language(
477
+ (int) filter_input( INPUT_POST, 'post_ID', FILTER_SANITIZE_STRING ),
478
+ 'locale'
479
  );
480
  }
481
  if ( isset( $_POST['pll_post_id'] ) ) {
482
  $pll_get_post_language = pll_get_post_language(
483
+ (int) filter_input( INPUT_POST, 'pll_post_id', FILTER_SANITIZE_STRING ),
484
+ 'locale'
485
  );
486
  }
487
  if ( isset( $_GET['post'] ) ) {
488
  $pll_get_post_language = pll_get_post_language(
489
+ (int) filter_input( INPUT_GET, 'post', FILTER_SANITIZE_STRING ),
490
+ 'locale'
491
  );
492
  }
493
  // phpcs:enable WordPress.Security.NonceVerification.Recommended
515
  );
516
 
517
  if ( $pll_get_language ) {
518
+ $pll_get_term_language = $pll_get_language->locale;
519
  }
520
  }
521
  // phpcs:enable WordPress.Security.NonceVerification.Missing
{classes → src/php}/class-requirements.php RENAMED
@@ -7,6 +7,7 @@
7
 
8
  namespace Cyr_To_Lat;
9
 
 
10
  use WP_Filesystem_Direct;
11
 
12
  if ( ! class_exists( __NAMESPACE__ . '\Requirements' ) ) {
@@ -16,6 +17,13 @@ if ( ! class_exists( __NAMESPACE__ . '\Requirements' ) ) {
16
  */
17
  class Requirements {
18
 
 
 
 
 
 
 
 
19
  /**
20
  * Admin notices.
21
  *
@@ -31,27 +39,24 @@ if ( ! class_exists( __NAMESPACE__ . '\Requirements' ) ) {
31
  protected $wp_filesystem;
32
 
33
  /**
34
- * Restrict notice to Cyr To Lat settings page.
35
  *
36
  * @var array
37
  */
38
- protected $cyr2lat_page;
39
 
40
  /**
41
  * Requirements constructor.
42
  *
 
43
  * @param Admin_Notices $admin_notices Admin notices.
44
  * @param WP_Filesystem_Direct $wp_filesystem File system.
45
- *
46
- * @noinspection PhpIncludeInspection
47
  */
48
- public function __construct( $admin_notices = null, $wp_filesystem = null ) {
49
- $this->admin_notices = $admin_notices;
50
- if ( ! $this->admin_notices ) {
51
- $this->admin_notices = new Admin_Notices();
52
- }
53
 
54
- $this->cyr2lat_page = [ 'page' => Settings::SCREEN_ID ];
55
 
56
  // @codeCoverageIgnoreStart
57
  if ( ! function_exists( 'WP_Filesystem' ) ) {
@@ -136,7 +141,7 @@ if ( ! class_exists( __NAMESPACE__ . '\Requirements' ) ) {
136
  $this->admin_notices->add_notice(
137
  __( 'Unable to get filesystem access.', 'cyr2lat' ),
138
  'notice notice-error',
139
- $this->cyr2lat_page
140
  );
141
  $this->ask_to_increase_max_input_vars();
142
 
@@ -160,7 +165,7 @@ if ( ! class_exists( __NAMESPACE__ . '\Requirements' ) ) {
160
  /* translators: 1: Wait time in seconds */
161
  $message .= sprintf( __( 'Please try again in %d s.', 'cyr2lat' ), $time_left );
162
 
163
- $this->admin_notices->add_notice( $message, 'notice notice-error', $this->cyr2lat_page );
164
  } else {
165
  $this->ask_to_increase_max_input_vars();
166
  }
@@ -231,7 +236,7 @@ if ( ! class_exists( __NAMESPACE__ . '\Requirements' ) ) {
231
  $this->admin_notices->add_notice(
232
  $message,
233
  'notice notice-error',
234
- [ 'page' => Settings::SCREEN_ID ]
235
  );
236
  }
237
  }
7
 
8
  namespace Cyr_To_Lat;
9
 
10
+ use Cyr_To_Lat\Settings\Settings;
11
  use WP_Filesystem_Direct;
12
 
13
  if ( ! class_exists( __NAMESPACE__ . '\Requirements' ) ) {
17
  */
18
  class Requirements {
19
 
20
+ /**
21
+ * Settings.
22
+ *
23
+ * @var Settings
24
+ */
25
+ protected $settings;
26
+
27
  /**
28
  * Admin notices.
29
  *
39
  protected $wp_filesystem;
40
 
41
  /**
42
+ * Restrict notices to Cyr To Lat settings admin screens.
43
  *
44
  * @var array
45
  */
46
+ protected $screen_ids;
47
 
48
  /**
49
  * Requirements constructor.
50
  *
51
+ * @param Settings $settings Settings.
52
  * @param Admin_Notices $admin_notices Admin notices.
53
  * @param WP_Filesystem_Direct $wp_filesystem File system.
 
 
54
  */
55
+ public function __construct( $settings, $admin_notices, $wp_filesystem = null ) {
56
+ $this->settings = $settings;
57
+ $this->screen_ids = [ 'screen_ids' => $this->settings->screen_ids() ];
 
 
58
 
59
+ $this->admin_notices = $admin_notices;
60
 
61
  // @codeCoverageIgnoreStart
62
  if ( ! function_exists( 'WP_Filesystem' ) ) {
141
  $this->admin_notices->add_notice(
142
  __( 'Unable to get filesystem access.', 'cyr2lat' ),
143
  'notice notice-error',
144
+ $this->screen_ids
145
  );
146
  $this->ask_to_increase_max_input_vars();
147
 
165
  /* translators: 1: Wait time in seconds */
166
  $message .= sprintf( __( 'Please try again in %d s.', 'cyr2lat' ), $time_left );
167
 
168
+ $this->admin_notices->add_notice( $message, 'notice notice-error', $this->screen_ids );
169
  } else {
170
  $this->ask_to_increase_max_input_vars();
171
  }
236
  $this->admin_notices->add_notice(
237
  $message,
238
  'notice notice-error',
239
+ [ 'screen_ids' => $this->settings->screen_ids() ]
240
  );
241
  }
242
  }
{classes → src/php}/class-wp-cli.php RENAMED
File without changes
vendor/autoload.php CHANGED
@@ -4,4 +4,4 @@
4
 
5
  require_once __DIR__ . '/composer/autoload_real.php';
6
 
7
- return ComposerAutoloaderInitab59197479483050e615845d183c2e8c::getLoader();
4
 
5
  require_once __DIR__ . '/composer/autoload_real.php';
6
 
7
+ return ComposerAutoloaderInitedbf0ae69d823c20f1f2c9984482bd8e::getLoader();
vendor/composer/ClassLoader.php CHANGED
@@ -311,8 +311,10 @@ class ClassLoader
311
  spl_autoload_register(array($this, 'loadClass'), true, $prepend);
312
 
313
  if (null === $this->vendorDir) {
314
- //no-op
315
- } elseif ($prepend) {
 
 
316
  self::$registeredLoaders = array($this->vendorDir => $this) + self::$registeredLoaders;
317
  } else {
318
  unset(self::$registeredLoaders[$this->vendorDir]);
311
  spl_autoload_register(array($this, 'loadClass'), true, $prepend);
312
 
313
  if (null === $this->vendorDir) {
314
+ return;
315
+ }
316
+
317
+ if ($prepend) {
318
  self::$registeredLoaders = array($this->vendorDir => $this) + self::$registeredLoaders;
319
  } else {
320
  unset(self::$registeredLoaders[$this->vendorDir]);
vendor/composer/InstalledVersions.php CHANGED
@@ -25,12 +25,12 @@ class InstalledVersions
25
  private static $installed = array (
26
  'root' =>
27
  array (
28
- 'pretty_version' => '4.6.4',
29
- 'version' => '4.6.4.0',
30
  'aliases' =>
31
  array (
32
  ),
33
- 'reference' => '4117c3e807c8bed4120639cfcd1ef7f34cb2a543',
34
  'name' => 'mihdan/cyr2lat',
35
  ),
36
  'versions' =>
@@ -46,12 +46,12 @@ private static $installed = array (
46
  ),
47
  'mihdan/cyr2lat' =>
48
  array (
49
- 'pretty_version' => '4.6.4',
50
- 'version' => '4.6.4.0',
51
  'aliases' =>
52
  array (
53
  ),
54
- 'reference' => '4117c3e807c8bed4120639cfcd1ef7f34cb2a543',
55
  ),
56
  'roundcube/plugin-installer' =>
57
  array (
@@ -290,7 +290,6 @@ self::$canGetVendors = method_exists('Composer\Autoload\ClassLoader', 'getRegist
290
  $installed = array();
291
 
292
  if (self::$canGetVendors) {
293
-
294
  foreach (ClassLoader::getRegisteredLoaders() as $vendorDir => $loader) {
295
  if (isset(self::$installedByVendor[$vendorDir])) {
296
  $installed[] = self::$installedByVendor[$vendorDir];
25
  private static $installed = array (
26
  'root' =>
27
  array (
28
+ 'pretty_version' => '5.0.0',
29
+ 'version' => '5.0.0.0',
30
  'aliases' =>
31
  array (
32
  ),
33
+ 'reference' => 'a05290b03db0f9d0bddd9b489170e0e867b89b28',
34
  'name' => 'mihdan/cyr2lat',
35
  ),
36
  'versions' =>
46
  ),
47
  'mihdan/cyr2lat' =>
48
  array (
49
+ 'pretty_version' => '5.0.0',
50
+ 'version' => '5.0.0.0',
51
  'aliases' =>
52
  array (
53
  ),
54
+ 'reference' => 'a05290b03db0f9d0bddd9b489170e0e867b89b28',
55
  ),
56
  'roundcube/plugin-installer' =>
57
  array (
290
  $installed = array();
291
 
292
  if (self::$canGetVendors) {
 
293
  foreach (ClassLoader::getRegisteredLoaders() as $vendorDir => $loader) {
294
  if (isset(self::$installedByVendor[$vendorDir])) {
295
  $installed[] = self::$installedByVendor[$vendorDir];
vendor/composer/autoload_classmap.php CHANGED
@@ -7,18 +7,23 @@ $baseDir = dirname($vendorDir);
7
 
8
  return array(
9
  'Composer\\InstalledVersions' => $vendorDir . '/composer/InstalledVersions.php',
10
- 'Cyr_To_Lat\\ACF' => $baseDir . '/classes/class-acf.php',
11
- 'Cyr_To_Lat\\Admin_Notices' => $baseDir . '/classes/class-admin-notices.php',
12
- 'Cyr_To_Lat\\Conversion_Process' => $baseDir . '/classes/background-processes/class-conversion-process.php',
13
- 'Cyr_To_Lat\\Conversion_Tables' => $baseDir . '/classes/class-conversion-tables.php',
14
- 'Cyr_To_Lat\\Converter' => $baseDir . '/classes/class-converter.php',
15
  'Cyr_To_Lat\\KAGG\\WP_Background_Processing\\WP_Async_Request' => $baseDir . '/lib/wp-background-processing/class-wp-async-request.php',
16
  'Cyr_To_Lat\\KAGG\\WP_Background_Processing\\WP_Background_Process' => $baseDir . '/lib/wp-background-processing/class-wp-background-process.php',
17
- 'Cyr_To_Lat\\Main' => $baseDir . '/classes/class-main.php',
18
- 'Cyr_To_Lat\\Post_Conversion_Process' => $baseDir . '/classes/background-processes/class-post-conversion-process.php',
19
- 'Cyr_To_Lat\\Requirements' => $baseDir . '/classes/class-requirements.php',
20
- 'Cyr_To_Lat\\Settings' => $baseDir . '/classes/class-settings.php',
 
 
 
 
 
21
  'Cyr_To_Lat\\Symfony\\Polyfill\\Mbstring\\Mbstring' => $baseDir . '/lib/polyfill-mbstring/Mbstring.php',
22
- 'Cyr_To_Lat\\Term_Conversion_Process' => $baseDir . '/classes/background-processes/class-term-conversion-process.php',
23
- 'Cyr_To_Lat\\WP_CLI' => $baseDir . '/classes/class-wp-cli.php',
24
  );
7
 
8
  return array(
9
  'Composer\\InstalledVersions' => $vendorDir . '/composer/InstalledVersions.php',
10
+ 'Cyr_To_Lat\\ACF' => $baseDir . '/src/php/class-acf.php',
11
+ 'Cyr_To_Lat\\Admin_Notices' => $baseDir . '/src/php/class-admin-notices.php',
12
+ 'Cyr_To_Lat\\Conversion_Process' => $baseDir . '/src/php/background-processes/class-conversion-process.php',
13
+ 'Cyr_To_Lat\\Conversion_Tables' => $baseDir . '/src/php/class-conversion-tables.php',
14
+ 'Cyr_To_Lat\\Converter' => $baseDir . '/src/php/class-converter.php',
15
  'Cyr_To_Lat\\KAGG\\WP_Background_Processing\\WP_Async_Request' => $baseDir . '/lib/wp-background-processing/class-wp-async-request.php',
16
  'Cyr_To_Lat\\KAGG\\WP_Background_Processing\\WP_Background_Process' => $baseDir . '/lib/wp-background-processing/class-wp-background-process.php',
17
+ 'Cyr_To_Lat\\Main' => $baseDir . '/src/php/class-main.php',
18
+ 'Cyr_To_Lat\\Post_Conversion_Process' => $baseDir . '/src/php/background-processes/class-post-conversion-process.php',
19
+ 'Cyr_To_Lat\\Requirements' => $baseDir . '/src/php/class-requirements.php',
20
+ 'Cyr_To_Lat\\Settings\\Abstracts\\SettingsBase' => $baseDir . '/src/php/Settings/Abstracts/SettingsBase.php',
21
+ 'Cyr_To_Lat\\Settings\\Abstracts\\SettingsInterface' => $baseDir . '/src/php/Settings/Abstracts/SettingsInterface.php',
22
+ 'Cyr_To_Lat\\Settings\\Converter' => $baseDir . '/src/php/Settings/Converter.php',
23
+ 'Cyr_To_Lat\\Settings\\PluginSettingsBase' => $baseDir . '/src/php/Settings/PluginSettingsBase.php',
24
+ 'Cyr_To_Lat\\Settings\\Settings' => $baseDir . '/src/php/Settings/Settings.php',
25
+ 'Cyr_To_Lat\\Settings\\Tables' => $baseDir . '/src/php/Settings/Tables.php',
26
  'Cyr_To_Lat\\Symfony\\Polyfill\\Mbstring\\Mbstring' => $baseDir . '/lib/polyfill-mbstring/Mbstring.php',
27
+ 'Cyr_To_Lat\\Term_Conversion_Process' => $baseDir . '/src/php/background-processes/class-term-conversion-process.php',
28
+ 'Cyr_To_Lat\\WP_CLI' => $baseDir . '/src/php/class-wp-cli.php',
29
  );
vendor/composer/autoload_real.php CHANGED
@@ -2,7 +2,7 @@
2
 
3
  // autoload_real.php @generated by Composer
4
 
5
- class ComposerAutoloaderInitab59197479483050e615845d183c2e8c
6
  {
7
  private static $loader;
8
 
@@ -22,15 +22,15 @@ class ComposerAutoloaderInitab59197479483050e615845d183c2e8c
22
  return self::$loader;
23
  }
24
 
25
- spl_autoload_register(array('ComposerAutoloaderInitab59197479483050e615845d183c2e8c', 'loadClassLoader'), true, true);
26
  self::$loader = $loader = new \Composer\Autoload\ClassLoader(\dirname(\dirname(__FILE__)));
27
- spl_autoload_unregister(array('ComposerAutoloaderInitab59197479483050e615845d183c2e8c', 'loadClassLoader'));
28
 
29
  $useStaticLoader = PHP_VERSION_ID >= 50600 && !defined('HHVM_VERSION') && (!function_exists('zend_loader_file_encoded') || !zend_loader_file_encoded());
30
  if ($useStaticLoader) {
31
  require __DIR__ . '/autoload_static.php';
32
 
33
- call_user_func(\Composer\Autoload\ComposerStaticInitab59197479483050e615845d183c2e8c::getInitializer($loader));
34
  } else {
35
  $map = require __DIR__ . '/autoload_namespaces.php';
36
  foreach ($map as $namespace => $path) {
@@ -51,19 +51,19 @@ class ComposerAutoloaderInitab59197479483050e615845d183c2e8c
51
  $loader->register(true);
52
 
53
  if ($useStaticLoader) {
54
- $includeFiles = Composer\Autoload\ComposerStaticInitab59197479483050e615845d183c2e8c::$files;
55
  } else {
56
  $includeFiles = require __DIR__ . '/autoload_files.php';
57
  }
58
  foreach ($includeFiles as $fileIdentifier => $file) {
59
- composerRequireab59197479483050e615845d183c2e8c($fileIdentifier, $file);
60
  }
61
 
62
  return $loader;
63
  }
64
  }
65
 
66
- function composerRequireab59197479483050e615845d183c2e8c($fileIdentifier, $file)
67
  {
68
  if (empty($GLOBALS['__composer_autoload_files'][$fileIdentifier])) {
69
  require $file;
2
 
3
  // autoload_real.php @generated by Composer
4
 
5
+ class ComposerAutoloaderInitedbf0ae69d823c20f1f2c9984482bd8e
6
  {
7
  private static $loader;
8
 
22
  return self::$loader;
23
  }
24
 
25
+ spl_autoload_register(array('ComposerAutoloaderInitedbf0ae69d823c20f1f2c9984482bd8e', 'loadClassLoader'), true, true);
26
  self::$loader = $loader = new \Composer\Autoload\ClassLoader(\dirname(\dirname(__FILE__)));
27
+ spl_autoload_unregister(array('ComposerAutoloaderInitedbf0ae69d823c20f1f2c9984482bd8e', 'loadClassLoader'));
28
 
29
  $useStaticLoader = PHP_VERSION_ID >= 50600 && !defined('HHVM_VERSION') && (!function_exists('zend_loader_file_encoded') || !zend_loader_file_encoded());
30
  if ($useStaticLoader) {
31
  require __DIR__ . '/autoload_static.php';
32
 
33
+ call_user_func(\Composer\Autoload\ComposerStaticInitedbf0ae69d823c20f1f2c9984482bd8e::getInitializer($loader));
34
  } else {
35
  $map = require __DIR__ . '/autoload_namespaces.php';
36
  foreach ($map as $namespace => $path) {
51
  $loader->register(true);
52
 
53
  if ($useStaticLoader) {
54
+ $includeFiles = Composer\Autoload\ComposerStaticInitedbf0ae69d823c20f1f2c9984482bd8e::$files;
55
  } else {
56
  $includeFiles = require __DIR__ . '/autoload_files.php';
57
  }
58
  foreach ($includeFiles as $fileIdentifier => $file) {
59
+ composerRequireedbf0ae69d823c20f1f2c9984482bd8e($fileIdentifier, $file);
60
  }
61
 
62
  return $loader;
63
  }
64
  }
65
 
66
+ function composerRequireedbf0ae69d823c20f1f2c9984482bd8e($fileIdentifier, $file)
67
  {
68
  if (empty($GLOBALS['__composer_autoload_files'][$fileIdentifier])) {
69
  require $file;
vendor/composer/autoload_static.php CHANGED
@@ -4,7 +4,7 @@
4
 
5
  namespace Composer\Autoload;
6
 
7
- class ComposerStaticInitab59197479483050e615845d183c2e8c
8
  {
9
  public static $files = array (
10
  '344a0f93a05b8ca362c22e39586db500' => __DIR__ . '/../..' . '/lib/polyfill-mbstring/bootstrap.php',
@@ -26,28 +26,33 @@ class ComposerStaticInitab59197479483050e615845d183c2e8c
26
 
27
  public static $classMap = array (
28
  'Composer\\InstalledVersions' => __DIR__ . '/..' . '/composer/InstalledVersions.php',
29
- 'Cyr_To_Lat\\ACF' => __DIR__ . '/../..' . '/classes/class-acf.php',
30
- 'Cyr_To_Lat\\Admin_Notices' => __DIR__ . '/../..' . '/classes/class-admin-notices.php',
31
- 'Cyr_To_Lat\\Conversion_Process' => __DIR__ . '/../..' . '/classes/background-processes/class-conversion-process.php',
32
- 'Cyr_To_Lat\\Conversion_Tables' => __DIR__ . '/../..' . '/classes/class-conversion-tables.php',
33
- 'Cyr_To_Lat\\Converter' => __DIR__ . '/../..' . '/classes/class-converter.php',
34
  'Cyr_To_Lat\\KAGG\\WP_Background_Processing\\WP_Async_Request' => __DIR__ . '/../..' . '/lib/wp-background-processing/class-wp-async-request.php',
35
  'Cyr_To_Lat\\KAGG\\WP_Background_Processing\\WP_Background_Process' => __DIR__ . '/../..' . '/lib/wp-background-processing/class-wp-background-process.php',
36
- 'Cyr_To_Lat\\Main' => __DIR__ . '/../..' . '/classes/class-main.php',
37
- 'Cyr_To_Lat\\Post_Conversion_Process' => __DIR__ . '/../..' . '/classes/background-processes/class-post-conversion-process.php',
38
- 'Cyr_To_Lat\\Requirements' => __DIR__ . '/../..' . '/classes/class-requirements.php',
39
- 'Cyr_To_Lat\\Settings' => __DIR__ . '/../..' . '/classes/class-settings.php',
 
 
 
 
 
40
  'Cyr_To_Lat\\Symfony\\Polyfill\\Mbstring\\Mbstring' => __DIR__ . '/../..' . '/lib/polyfill-mbstring/Mbstring.php',
41
- 'Cyr_To_Lat\\Term_Conversion_Process' => __DIR__ . '/../..' . '/classes/background-processes/class-term-conversion-process.php',
42
- 'Cyr_To_Lat\\WP_CLI' => __DIR__ . '/../..' . '/classes/class-wp-cli.php',
43
  );
44
 
45
  public static function getInitializer(ClassLoader $loader)
46
  {
47
  return \Closure::bind(function () use ($loader) {
48
- $loader->prefixLengthsPsr4 = ComposerStaticInitab59197479483050e615845d183c2e8c::$prefixLengthsPsr4;
49
- $loader->prefixDirsPsr4 = ComposerStaticInitab59197479483050e615845d183c2e8c::$prefixDirsPsr4;
50
- $loader->classMap = ComposerStaticInitab59197479483050e615845d183c2e8c::$classMap;
51
 
52
  }, null, ClassLoader::class);
53
  }
4
 
5
  namespace Composer\Autoload;
6
 
7
+ class ComposerStaticInitedbf0ae69d823c20f1f2c9984482bd8e
8
  {
9
  public static $files = array (
10
  '344a0f93a05b8ca362c22e39586db500' => __DIR__ . '/../..' . '/lib/polyfill-mbstring/bootstrap.php',
26
 
27
  public static $classMap = array (
28
  'Composer\\InstalledVersions' => __DIR__ . '/..' . '/composer/InstalledVersions.php',
29
+ 'Cyr_To_Lat\\ACF' => __DIR__ . '/../..' . '/src/php/class-acf.php',
30
+ 'Cyr_To_Lat\\Admin_Notices' => __DIR__ . '/../..' . '/src/php/class-admin-notices.php',
31
+ 'Cyr_To_Lat\\Conversion_Process' => __DIR__ . '/../..' . '/src/php/background-processes/class-conversion-process.php',
32
+ 'Cyr_To_Lat\\Conversion_Tables' => __DIR__ . '/../..' . '/src/php/class-conversion-tables.php',
33
+ 'Cyr_To_Lat\\Converter' => __DIR__ . '/../..' . '/src/php/class-converter.php',
34
  'Cyr_To_Lat\\KAGG\\WP_Background_Processing\\WP_Async_Request' => __DIR__ . '/../..' . '/lib/wp-background-processing/class-wp-async-request.php',
35
  'Cyr_To_Lat\\KAGG\\WP_Background_Processing\\WP_Background_Process' => __DIR__ . '/../..' . '/lib/wp-background-processing/class-wp-background-process.php',
36
+ 'Cyr_To_Lat\\Main' => __DIR__ . '/../..' . '/src/php/class-main.php',
37
+ 'Cyr_To_Lat\\Post_Conversion_Process' => __DIR__ . '/../..' . '/src/php/background-processes/class-post-conversion-process.php',
38
+ 'Cyr_To_Lat\\Requirements' => __DIR__ . '/../..' . '/src/php/class-requirements.php',
39
+ 'Cyr_To_Lat\\Settings\\Abstracts\\SettingsBase' => __DIR__ . '/../..' . '/src/php/Settings/Abstracts/SettingsBase.php',
40
+ 'Cyr_To_Lat\\Settings\\Abstracts\\SettingsInterface' => __DIR__ . '/../..' . '/src/php/Settings/Abstracts/SettingsInterface.php',
41
+ 'Cyr_To_Lat\\Settings\\Converter' => __DIR__ . '/../..' . '/src/php/Settings/Converter.php',
42
+ 'Cyr_To_Lat\\Settings\\PluginSettingsBase' => __DIR__ . '/../..' . '/src/php/Settings/PluginSettingsBase.php',
43
+ 'Cyr_To_Lat\\Settings\\Settings' => __DIR__ . '/../..' . '/src/php/Settings/Settings.php',
44
+ 'Cyr_To_Lat\\Settings\\Tables' => __DIR__ . '/../..' . '/src/php/Settings/Tables.php',
45
  'Cyr_To_Lat\\Symfony\\Polyfill\\Mbstring\\Mbstring' => __DIR__ . '/../..' . '/lib/polyfill-mbstring/Mbstring.php',
46
+ 'Cyr_To_Lat\\Term_Conversion_Process' => __DIR__ . '/../..' . '/src/php/background-processes/class-term-conversion-process.php',
47
+ 'Cyr_To_Lat\\WP_CLI' => __DIR__ . '/../..' . '/src/php/class-wp-cli.php',
48
  );
49
 
50
  public static function getInitializer(ClassLoader $loader)
51
  {
52
  return \Closure::bind(function () use ($loader) {
53
+ $loader->prefixLengthsPsr4 = ComposerStaticInitedbf0ae69d823c20f1f2c9984482bd8e::$prefixLengthsPsr4;
54
+ $loader->prefixDirsPsr4 = ComposerStaticInitedbf0ae69d823c20f1f2c9984482bd8e::$prefixDirsPsr4;
55
+ $loader->classMap = ComposerStaticInitedbf0ae69d823c20f1f2c9984482bd8e::$classMap;
56
 
57
  }, null, ClassLoader::class);
58
  }
vendor/composer/installed.php CHANGED
@@ -1,12 +1,12 @@
1
  <?php return array (
2
  'root' =>
3
  array (
4
- 'pretty_version' => '4.6.4',
5
- 'version' => '4.6.4.0',
6
  'aliases' =>
7
  array (
8
  ),
9
- 'reference' => '4117c3e807c8bed4120639cfcd1ef7f34cb2a543',
10
  'name' => 'mihdan/cyr2lat',
11
  ),
12
  'versions' =>
@@ -22,12 +22,12 @@
22
  ),
23
  'mihdan/cyr2lat' =>
24
  array (
25
- 'pretty_version' => '4.6.4',
26
- 'version' => '4.6.4.0',
27
  'aliases' =>
28
  array (
29
  ),
30
- 'reference' => '4117c3e807c8bed4120639cfcd1ef7f34cb2a543',
31
  ),
32
  'roundcube/plugin-installer' =>
33
  array (
1
  <?php return array (
2
  'root' =>
3
  array (
4
+ 'pretty_version' => '5.0.0',
5
+ 'version' => '5.0.0.0',
6
  'aliases' =>
7
  array (
8
  ),
9
+ 'reference' => 'a05290b03db0f9d0bddd9b489170e0e867b89b28',
10
  'name' => 'mihdan/cyr2lat',
11
  ),
12
  'versions' =>
22
  ),
23
  'mihdan/cyr2lat' =>
24
  array (
25
+ 'pretty_version' => '5.0.0',
26
+ 'version' => '5.0.0.0',
27
  'aliases' =>
28
  array (
29
  ),
30
+ 'reference' => 'a05290b03db0f9d0bddd9b489170e0e867b89b28',
31
  ),
32
  'roundcube/plugin-installer' =>
33
  array (