diff --git a/extension.json b/extension.json
index d56d18a..f4dc2b8 100644
--- a/extension.json
+++ b/extension.json
@@ -223,6 +223,19 @@
"isekai-offcanvastoc-menubutton",
"isekai-offcanvastoc-description-item"
]
+ },
+ "ext.isekai.masonry": {
+ "scripts": [
+ "masonry/ext.isekai.masonry.js",
+ "masonry/ext.isekai.masonry.base.js"
+ ],
+ "styles": [
+ "masonry/ext.isekai.masonry.less"
+ ],
+ "targets": [
+ "desktop",
+ "mobile"
+ ]
}
},
"ResourceFileModulePaths": {
@@ -236,7 +249,9 @@
"exfont": "text/mediawiki",
"details": "text/mediawiki",
"summary": "text/mediawiki",
- "veval": "text/mediawiki"
+ "veval": "text/mediawiki",
+ "masonry": "text/mediawiki",
+ "masonryitem": "text/mediawiki"
}
}
},
diff --git a/includes/ExtraFontWidget.php b/includes/ExtraFontWidget.php
index 6855e7f..d0ef960 100644
--- a/includes/ExtraFontWidget.php
+++ b/includes/ExtraFontWidget.php
@@ -8,7 +8,7 @@ class ExtraFontWidget {
public static function create($text, $params, \Parser $parser, \PPFrame $frame) {
$existsFonts = $parser->extIsekaiWidgetsCache->get('extraFonts', INF, []);
- $content = $text = $parser->recursiveTagParse($text, $frame);
+ $content = $text = $parser->recursiveTagParseFully($text, $frame);
if (empty($params['name'])) {
return '' . wfMessage('isekai-font-error-invalid-params')->parse() . '' . $content;
}
diff --git a/includes/MasonryItemWidget.php b/includes/MasonryItemWidget.php
new file mode 100644
index 0000000..22025dc
--- /dev/null
+++ b/includes/MasonryItemWidget.php
@@ -0,0 +1,69 @@
+recursiveTagParse($text, $frame);
+ if (!$masonryParams) {
+ return '' . wfMessage('isekai-masonry-item-must-in-masonry')->parse() . '' . $content;
+ }
+
+ $maxCol = $masonryParams['cols'];
+
+ $className = ['isekai-masonry-item'];
+
+ if (isset($params['col'])) {
+ $value = $params['col'];
+ $col = $value === 'full' ? $maxCol : min($maxCol, intval($value));
+ $className[] = 'col-' . $col;
+ } else {
+ $className[] = 'col-1';
+ }
+
+ if (isset($params['xs-col'])) {
+ $value = $params['xs-col'];
+ $col = $value === 'full' ? $maxCol : min($maxCol, intval($value));
+ $className[] = 'col-xs-' . $col;
+ }
+
+ if (isset($params['sm-col'])) {
+ $value = $params['sm-col'];
+ $col = $value === 'full' ? $maxCol : min($maxCol, intval($value));
+ $className[] = 'col-sm-' . $col;
+ }
+
+ if (isset($params['md-col'])) {
+ $value = $params['md-col'];
+ $col = $value === 'full' ? $maxCol : min($maxCol, intval($value));
+ $className[] = 'col-md-' . $col;
+ }
+
+ if (isset($params['lg-col'])) {
+ $value = $params['lg-col'];
+ $col = $value === 'full' ? $maxCol : min($maxCol, intval($value));
+ $className[] = 'col-lg-' . $col;
+ }
+
+ if (isset($params['xl-col'])) {
+ $value = $params['xl-col'];
+ $col = $value === 'full' ? $maxCol : min($maxCol, intval($value));
+ $className[] = 'col-xl-' . $col;
+ }
+
+ return Html::openElement('div', [
+ 'class' => implode(' ', $className),
+ ]) . $content . Html::closeElement('div');
+ }
+}
\ No newline at end of file
diff --git a/includes/MasonryWidget.php b/includes/MasonryWidget.php
new file mode 100644
index 0000000..0817931
--- /dev/null
+++ b/includes/MasonryWidget.php
@@ -0,0 +1,100 @@
+ 576,
+ 'sm' => 768,
+ 'md' => 992,
+ 'lg' => 1200,
+ 'xl' => 1400,
+ ];
+
+ foreach ($breakpoints as $breakpoint => $width) {
+ $paramKey = "{$breakpoint}-cols";
+ if (!isset($params[$paramKey])) {
+ continue;
+ }
+
+ $maxCol = max(1, $params[$paramKey]);
+ $colWidth = round(100 / $maxCol, 6);
+
+ $css .= <<getOutput()->addModules(['ext.isekai.masonry']);
+
+ $params['cols'] = intval($params['cols'] ?? 2);
+ $params['xs-cols'] = intval($params['xs-cols'] ?? 2);
+ $params['sm-cols'] = intval($params['sm-cols'] ?? 2);
+ $params['md-cols'] = intval($params['md-cols'] ?? 2);
+ $params['lg-cols'] = intval($params['lg-cols'] ?? 2);
+ $params['xl-cols'] = intval($params['xl-cols'] ?? 3);
+ $params['gutter'] = intval($params['gutter'] ?? 10);
+
+ $className = ['isekai-masonry', "max-col-{$params['cols']}"];
+
+ self::$paramsStack[] = $params;
+ $content = $parser->recursiveTagParseFully($text, $frame);
+ array_pop(self::$paramsStack);
+
+ return [
+ self::getStyle($params) .
+ Html::openElement('div', [
+ 'class' => implode(' ', $className),
+ 'data-cols' => $params['cols'],
+ 'data-xs-cols' => $params['xs-cols'],
+ 'data-sm-cols' => $params['sm-cols'],
+ 'data-md-cols' => $params['md-cols'],
+ 'data-lg-cols' => $params['lg-cols'],
+ 'data-xl-cols' => $params['xl-cols'],
+ 'data-gutter' => $params['gutter'],
+ ]) .
+ Html::element('div', [
+ 'class' => 'isekai-masonry-sizer',
+ ]) .
+ Html::element('div', [
+ 'class' => 'isekai-masonry-gutter-sizer',
+ ]) .
+ $content .
+ Html::closeElement('div'),
+ "markerType" => 'nowiki'
+ ];
+ }
+}
\ No newline at end of file
diff --git a/includes/Widgets.php b/includes/Widgets.php
index f34d691..f9bc027 100644
--- a/includes/Widgets.php
+++ b/includes/Widgets.php
@@ -34,6 +34,9 @@ class Widgets {
$parser->setHook('veval', [VEvalWidget::class, 'create']);
+ $parser->setHook('masonry', [MasonryWidget::class, 'create']);
+ $parser->setHook('masonryitem', [MasonryItemWidget::class, 'create']);
+
return true;
}
diff --git a/modules/masonry/ext.isekai.masonry.base.js b/modules/masonry/ext.isekai.masonry.base.js
new file mode 100644
index 0000000..fe9ca8e
--- /dev/null
+++ b/modules/masonry/ext.isekai.masonry.base.js
@@ -0,0 +1,9 @@
+$(function () {
+ console.log('IsekaiWidgets: Masonry extension loaded');
+ new isekai.lib.Masonry('.isekai-masonry', {
+ itemSelector: '.isekai-masonry-item',
+ columnWidth: '.isekai-masonry-sizer',
+ gutter: '.isekai-masonry-gutter-sizer',
+ percentPosition: true
+ });
+});
\ No newline at end of file
diff --git a/modules/masonry/ext.isekai.masonry.js b/modules/masonry/ext.isekai.masonry.js
new file mode 100644
index 0000000..17a6506
--- /dev/null
+++ b/modules/masonry/ext.isekai.masonry.js
@@ -0,0 +1,2 @@
+/*! For license information please see ext.isekai.masonry.js.LICENSE.txt */
+(()=>{var t={388:(t,i,e)=>{var n,o;!function(s,r){"use strict";void 0===(o="function"==typeof(n=r)?n.call(i,e,i,t):n)||(t.exports=o)}(window,(function(){"use strict";var t=function(){var t=window.Element.prototype;if(t.matches)return"matches";if(t.matchesSelector)return"matchesSelector";for(var i=["webkit","moz","ms","o"],e=0;e{var n,o;!function(s,r){n=[e(388)],o=function(t){return function(t,i){"use strict";var e={extend:function(t,i){for(var e in i)t[e]=i[e];return t},modulo:function(t,i){return(t%i+i)%i}},n=Array.prototype.slice;e.makeArray=function(t){return Array.isArray(t)?t:null==t?[]:"object"==typeof t&&"number"==typeof t.length?n.call(t):[t]},e.removeFrom=function(t,i){var e=t.indexOf(i);-1!=e&&t.splice(e,1)},e.getParent=function(t,e){for(;t.parentNode&&t!=document.body;)if(t=t.parentNode,i(t,e))return t},e.getQueryElement=function(t){return"string"==typeof t?document.querySelector(t):t},e.handleEvent=function(t){var i="on"+t.type;this[i]&&this[i](t)},e.filterFindElements=function(t,n){t=e.makeArray(t);var o=[];return t.forEach((function(t){if(t instanceof HTMLElement)if(n){i(t,n)&&o.push(t);for(var e=t.querySelectorAll(n),s=0;s{var n,o;window,void 0===(o="function"==typeof(n=function(){"use strict";function t(t){var i=parseFloat(t);return-1==t.indexOf("%")&&!isNaN(i)&&i}var i="undefined"==typeof console?function(){}:function(t){console.error(t)},e=["paddingLeft","paddingRight","paddingTop","paddingBottom","marginLeft","marginRight","marginTop","marginBottom","borderLeftWidth","borderRightWidth","borderTopWidth","borderBottomWidth"],n=e.length;function o(t){var e=getComputedStyle(t);return e||i("Style returned "+e+". Are you running this code in a hidden iframe on Firefox? See https://bit.ly/getsizebug1"),e}var s,r=!1;function a(i){if(function(){if(!r){r=!0;var i=document.createElement("div");i.style.width="200px",i.style.padding="1px 2px 3px 4px",i.style.borderStyle="solid",i.style.borderWidth="1px 2px 3px 4px",i.style.boxSizing="border-box";var e=document.body||document.documentElement;e.appendChild(i);var n=o(i);s=200==Math.round(t(n.width)),a.isBoxSizeOuter=s,e.removeChild(i)}}(),"string"==typeof i&&(i=document.querySelector(i)),i&&"object"==typeof i&&i.nodeType){var h=o(i);if("none"==h.display)return function(){for(var t={width:0,height:0,innerWidth:0,innerHeight:0,outerWidth:0,outerHeight:0},i=0;i{var n,o,s;window,o=[e(831),e(232)],void 0===(s="function"==typeof(n=function(t,i){"use strict";var e=t.create("masonry");e.compatOptions.fitWidth="isFitWidth";var n=e.prototype;return n._resetLayout=function(){this.getSize(),this._getMeasurement("columnWidth","outerWidth"),this._getMeasurement("gutter","outerWidth"),this.measureColumns(),this.colYs=[];for(var t=0;t1&&e+t>this.cols?0:e;var n=i.size.outerWidth&&i.size.outerHeight;return this.horizontalColIndex=n?e+t:this.horizontalColIndex,{col:e,y:this._getColGroupY(e,t)}},n._manageStamp=function(t){var e=i(t),n=this._getElementOffset(t),o=this._getOption("originLeft")?n.left:n.right,s=o+e.outerWidth,r=Math.floor(o/this.columnWidth);r=Math.max(0,r);var a=Math.floor(s/this.columnWidth);a-=s%this.columnWidth?0:1,a=Math.min(this.cols-1,a);for(var h=(this._getOption("originTop")?n.top:n.bottom)+e.outerHeight,u=r;u<=a;u++)this.colYs[u]=Math.max(h,this.colYs[u])},n._getContainerSize=function(){this.maxY=Math.max.apply(Math,this.colYs);var t={height:this.maxY};return this._getOption("fitWidth")&&(t.width=this._getContainerFitWidth()),t},n._getContainerFitWidth=function(){for(var t=0,i=this.cols;--i&&0===this.colYs[i];)t++;return(this.cols-t)*this.columnWidth-this.gutter},n.needsResizeLayout=function(){var t=this.containerWidth;return this.getContainerWidth(),t!=this.containerWidth},e})?n.apply(i,o):n)||(t.exports=s)},818:(t,i,e)=>{var n,o,s;window,o=[e(285),e(232)],void 0===(s="function"==typeof(n=function(t,i){"use strict";var e=document.documentElement.style,n="string"==typeof e.transition?"transition":"WebkitTransition",o="string"==typeof e.transform?"transform":"WebkitTransform",s={WebkitTransition:"webkitTransitionEnd",transition:"transitionend"}[n],r={transform:o,transition:n,transitionDuration:n+"Duration",transitionProperty:n+"Property",transitionDelay:n+"Delay"};function a(t,i){t&&(this.element=t,this.layout=i,this.position={x:0,y:0},this._create())}var h=a.prototype=Object.create(t.prototype);h.constructor=a,h._create=function(){this._transn={ingProperties:{},clean:{},onEnd:{}},this.css({position:"absolute"})},h.handleEvent=function(t){var i="on"+t.type;this[i]&&this[i](t)},h.getSize=function(){this.size=i(this.element)},h.css=function(t){var i=this.element.style;for(var e in t)i[r[e]||e]=t[e]},h.getPosition=function(){var t=getComputedStyle(this.element),i=this.layout._getOption("originLeft"),e=this.layout._getOption("originTop"),n=t[i?"left":"right"],o=t[e?"top":"bottom"],s=parseFloat(n),r=parseFloat(o),a=this.layout.size;-1!=n.indexOf("%")&&(s=s/100*a.width),-1!=o.indexOf("%")&&(r=r/100*a.height),s=isNaN(s)?0:s,r=isNaN(r)?0:r,s-=i?a.paddingLeft:a.paddingRight,r-=e?a.paddingTop:a.paddingBottom,this.position.x=s,this.position.y=r},h.layoutPosition=function(){var t=this.layout.size,i={},e=this.layout._getOption("originLeft"),n=this.layout._getOption("originTop"),o=e?"paddingLeft":"paddingRight",s=e?"left":"right",r=e?"right":"left",a=this.position.x+t[o];i[s]=this.getXValue(a),i[r]="";var h=n?"paddingTop":"paddingBottom",u=n?"top":"bottom",l=n?"bottom":"top",d=this.position.y+t[h];i[u]=this.getYValue(d),i[l]="",this.css(i),this.emitEvent("layout",[this])},h.getXValue=function(t){var i=this.layout._getOption("horizontal");return this.layout.options.percentPosition&&!i?t/this.layout.size.width*100+"%":t+"px"},h.getYValue=function(t){var i=this.layout._getOption("horizontal");return this.layout.options.percentPosition&&i?t/this.layout.size.height*100+"%":t+"px"},h._transitionTo=function(t,i){this.getPosition();var e=this.position.x,n=this.position.y,o=t==this.position.x&&i==this.position.y;if(this.setPosition(t,i),!o||this.isTransitioning){var s=t-e,r=i-n,a={};a.transform=this.getTranslate(s,r),this.transition({to:a,onTransitionEnd:{transform:this.layoutPosition},isCleaning:!0})}else this.layoutPosition()},h.getTranslate=function(t,i){return"translate3d("+(t=this.layout._getOption("originLeft")?t:-t)+"px, "+(i=this.layout._getOption("originTop")?i:-i)+"px, 0)"},h.goTo=function(t,i){this.setPosition(t,i),this.layoutPosition()},h.moveTo=h._transitionTo,h.setPosition=function(t,i){this.position.x=parseFloat(t),this.position.y=parseFloat(i)},h._nonTransition=function(t){for(var i in this.css(t.to),t.isCleaning&&this._removeStyles(t.to),t.onTransitionEnd)t.onTransitionEnd[i].call(this)},h.transition=function(t){if(parseFloat(this.layout.options.transitionDuration)){var i=this._transn;for(var e in t.onTransitionEnd)i.onEnd[e]=t.onTransitionEnd[e];for(e in t.to)i.ingProperties[e]=!0,t.isCleaning&&(i.clean[e]=!0);t.from&&(this.css(t.from),this.element.offsetHeight),this.enableTransition(t.to),this.css(t.to),this.isTransitioning=!0}else this._nonTransition(t)};var u="opacity,"+o.replace(/([A-Z])/g,(function(t){return"-"+t.toLowerCase()}));h.enableTransition=function(){if(!this.isTransitioning){var t=this.layout.options.transitionDuration;t="number"==typeof t?t+"ms":t,this.css({transitionProperty:u,transitionDuration:t,transitionDelay:this.staggerDelay||0}),this.element.addEventListener(s,this,!1)}},h.onwebkitTransitionEnd=function(t){this.ontransitionend(t)},h.onotransitionend=function(t){this.ontransitionend(t)};var l={"-webkit-transform":"transform"};h.ontransitionend=function(t){if(t.target===this.element){var i=this._transn,e=l[t.propertyName]||t.propertyName;delete i.ingProperties[e],function(t){for(var i in t)return!1;return!0}(i.ingProperties)&&this.disableTransition(),e in i.clean&&(this.element.style[t.propertyName]="",delete i.clean[e]),e in i.onEnd&&(i.onEnd[e].call(this),delete i.onEnd[e]),this.emitEvent("transitionEnd",[this])}},h.disableTransition=function(){this.removeTransitionStyles(),this.element.removeEventListener(s,this,!1),this.isTransitioning=!1},h._removeStyles=function(t){var i={};for(var e in t)i[e]="";this.css(i)};var d={transitionProperty:"",transitionDuration:"",transitionDelay:""};return h.removeTransitionStyles=function(){this.css(d)},h.stagger=function(t){t=isNaN(t)?0:t,this.staggerDelay=t+"ms"},h.removeElem=function(){this.element.parentNode.removeChild(this.element),this.css({display:""}),this.emitEvent("remove",[this])},h.remove=function(){n&&parseFloat(this.layout.options.transitionDuration)?(this.once("transitionEnd",(function(){this.removeElem()})),this.hide()):this.removeElem()},h.reveal=function(){delete this.isHidden,this.css({display:""});var t=this.layout.options,i={};i[this.getHideRevealTransitionEndProperty("visibleStyle")]=this.onRevealTransitionEnd,this.transition({from:t.hiddenStyle,to:t.visibleStyle,isCleaning:!0,onTransitionEnd:i})},h.onRevealTransitionEnd=function(){this.isHidden||this.emitEvent("reveal")},h.getHideRevealTransitionEndProperty=function(t){var i=this.layout.options[t];if(i.opacity)return"opacity";for(var e in i)return e},h.hide=function(){this.isHidden=!0,this.css({display:""});var t=this.layout.options,i={};i[this.getHideRevealTransitionEndProperty("hiddenStyle")]=this.onHideTransitionEnd,this.transition({from:t.visibleStyle,to:t.hiddenStyle,isCleaning:!0,onTransitionEnd:i})},h.onHideTransitionEnd=function(){this.isHidden&&(this.css({display:"none"}),this.emitEvent("hide"))},h.destroy=function(){this.css({position:"",left:"",right:"",top:"",bottom:"",transition:"",transform:""})},a})?n.apply(i,o):n)||(t.exports=s)},831:(t,i,e)=>{var n,o;!function(s,r){"use strict";n=[e(285),e(232),e(982),e(818)],o=function(t,i,e,n){return function(t,i,e,n,o){var s=t.console,r=t.jQuery,a=function(){},h=0,u={};function l(t,i){var e=n.getQueryElement(t);if(e){this.element=e,r&&(this.$element=r(this.element)),this.options=n.extend({},this.constructor.defaults),this.option(i);var o=++h;this.element.outlayerGUID=o,u[o]=this,this._create(),this._getOption("initLayout")&&this.layout()}else s&&s.error("Bad element for "+this.constructor.namespace+": "+(e||t))}l.namespace="outlayer",l.Item=o,l.defaults={containerStyle:{position:"relative"},initLayout:!0,originLeft:!0,originTop:!0,resize:!0,resizeContainer:!0,transitionDuration:"0.4s",hiddenStyle:{opacity:0,transform:"scale(0.001)"},visibleStyle:{opacity:1,transform:"scale(1)"}};var d=l.prototype;function c(t){function i(){t.apply(this,arguments)}return i.prototype=Object.create(t.prototype),i.prototype.constructor=i,i}n.extend(d,i.prototype),d.option=function(t){n.extend(this.options,t)},d._getOption=function(t){var i=this.constructor.compatOptions[t];return i&&void 0!==this.options[i]?this.options[i]:this.options[t]},l.compatOptions={initLayout:"isInitLayout",horizontal:"isHorizontal",layoutInstant:"isLayoutInstant",originLeft:"isOriginLeft",originTop:"isOriginTop",resize:"isResizeBound",resizeContainer:"isResizingContainer"},d._create=function(){this.reloadItems(),this.stamps=[],this.stamp(this.options.stamp),n.extend(this.element.style,this.options.containerStyle),this._getOption("resize")&&this.bindResize()},d.reloadItems=function(){this.items=this._itemize(this.element.children)},d._itemize=function(t){for(var i=this._filterFindItemElements(t),e=this.constructor.Item,n=[],o=0;o{var i=t&&t.__esModule?()=>t.default:()=>t;return e.d(i,{a:i}),i},e.d=(t,i)=>{for(var n in i)e.o(i,n)&&!e.o(t,n)&&Object.defineProperty(t,n,{enumerable:!0,get:i[n]})},e.o=(t,i)=>Object.prototype.hasOwnProperty.call(t,i),(()=>{"use strict";var t=e(752);!function(t,i){var e="lib.Masonry".split(".");"isekai"in window||(window.isekai={});for(var n=window.isekai,o=0;o