git / VotyMehjong.html
KEXEL's picture
1.1
50a7d53 verified
<!DOCTYPE html>
<html lang="pt-br">
<head>
<meta charset="UTF-8">
<title>VOTYMEHJONG</title>
<meta name="viewport" content="width=device-width, initial-scale=1" />
<meta name="description" content="VOTYMEHJONG GAME ADAPTÁVEL..." />
<meta name="mobile-web-app-capable" content="yes" />
<meta name="apple-mobile-web-app-capable" content="yes" />
<meta name="application-name" content="VOTYMEHJONG" />
<meta name="apple-mobile-web-app-title" content="VOTYMEHJONG" />
<meta name="msapplication-starturl" content="https://kexel-git.static.hf.space/VotyMehjong.html?Installe=votymehjong" />
<meta name="theme-color" content="#222" />
<meta name="msapplication-navbutton-color" content="#222" />
<meta name="apple-mobile-web-app-status-bar-style" content="black-translucent" />
<meta name="apple-touch-fullscreen" content="yes" />
<meta name="robots" content="index, follow" />
<meta name="googlebot" content="index, follow" />
<meta property="og:image" content="https://placehold.co/512/3b2d50/bdbac1.png?font=roboto&text=VOTY.MEHJONG" />
<link rel="icon" type="image/png" sizes="512x512" href="https://placehold.co/512/3b2d50/bdbac1.png?font=roboto&text=VOTY.MEHJONG" />
<link rel="apple-touch-icon" type="image/png" sizes="512x512" href="https://placehold.co/512/3b2d50/bdbac1.png?font=roboto&text=VOTY.MEHJONG" />
<link rel="icon" type="image/png" sizes="192x192" href="https://placehold.co/192/3b2d50/bdbac1.png?font=roboto&text=VOTY.MEHJONG" />
<link rel="apple-touch-icon" type="image/png" sizes="192x192" href="https://placehold.co/192/3b2d50/bdbac1.png?font=roboto&text=VOTY.MEHJONG" />
<link rel="icon" type="image/png" sizes="144x144" href="https://placehold.co/144/3b2d50/bdbac1.png?font=roboto&text=VOTY.MEHJONG" />
<link rel="apple-touch-icon" type="image/png" sizes="144x144" href="https://placehold.co/144/3b2d50/bdbac1.png?font=roboto&text=VOTY.MEHJONG" />
<link rel="icon" type="image/png" sizes="96x96" href="https://placehold.co/96/3b2d50/bdbac1.png?font=roboto&text=VOTY.MEHJONG" />
<link rel="apple-touch-icon" type="image/png" sizes="96x96" href="https://placehold.co/96/3b2d50/bdbac1.png?font=roboto&text=VOTY.MEHJONG" />
<link rel="icon" type="image/png" sizes="72x72" href="https://placehold.co/72/3b2d50/bdbac1.png?font=roboto&text=VOTY.MEHJONG" />
<link rel="apple-touch-icon" type="image/png" sizes="72x72" href="https://placehold.co/72/3b2d50/bdbac1.png?font=roboto&text=VOTY.MEHJONG" />
<link rel="icon" type="image/png" sizes="48x48" href="https://placehold.co/48/3b2d50/bdbac1.png?font=roboto&text=VOTY.MEHJONG" />
<link rel="apple-touch-icon" type="image/png" sizes="48x48" href="https://placehold.co/48/3b2d50/bdbac1.png?font=roboto&text=VOTY.MEHJONG" />
<link rel="icon" type="image/png" sizes="36x36" href="https://placehold.co/36/3b2d50/bdbac1.png?font=roboto&text=VOTY.MEHJONG" />
<link rel="apple-touch-icon" type="image/png" sizes="36x36" href="https://placehold.co/36/3b2d50/bdbac1.png?font=roboto&text=VOTY.MEHJONG" />
<script src="data:application/javascript;charset=utf-8;base64,var colobd="#2b3035c7",cololis="#726d3b",colotxt="#fff9";document.write('<style>.pace{pointer-events:none;user-select:none;z-index:9999999;position:fixed;top:0;left:0;right:0;bottom:0;height:100%;width:100%;background:'+colobd+';overflow:hidden}.pace::before{content:"CARREGANDO...";color:'+colotxt+';position:absolute;top:47%;left:50%;transform:translate(-50%,-50%)}.pace .pace-progress{box-sizing:border-box;transform:translate3d(0,0,0);max-width:100%;position:fixed;z-index:2000;display:block;position:absolute;bottom:50%;right:100%;height:5px;width:100%;background:'+cololis+'}.pace.pace-inactive{display:none}</style>');(function(){function o(t,e){return function(){return t.apply(e,arguments)}}var u,c,i,n,y,t,l,v,r,s,a,e,p,w,b,h,f,d,g,m,k,S,q,x,L,P,T,R,j,O,E,M,A,C,N,_,F,U,W,X,D,H,I,z,G,B=[].slice,J={}.hasOwnProperty,K=function(t,e){for(var n in e)J.call(e,n)&&(t[n]=e[n]);function r(){this.constructor=t}return r.prototype=e.prototype,t.prototype=new r,t.__super__=e.prototype,t},Q=[].indexOf||function(t){for(var e=0,n=this.length;e<n;e++)if(e in this&&this[e]===t)return e;return-1};function V(){}for(f={className:"",catchupTime:100,initialRate:.03,minTime:250,ghostTime:100,maxProgressPerFrame:20,easeFactor:1.25,startOnPageLoad:!0,restartOnPushState:!0,restartOnRequestAfter:500,target:"body",elements:{checkInterval:100,selectors:["body"]},eventLag:{minSamples:10,sampleCount:3,lagThreshold:3},ajax:{trackMethods:["GET"],trackWebSockets:!0,ignoreURLs:[]}},L=function(){var t;return null!=(t="undefined"!=typeof performance&&null!==performance&&"function"==typeof performance.now?performance.now():void 0)?t:+new Date},T=window.requestAnimationFrame||window.mozRequestAnimationFrame||window.webkitRequestAnimationFrame||window.msRequestAnimationFrame,h=window.cancelAnimationFrame||window.mozCancelAnimationFrame,a=function(t,e,n){return("function"==typeof t.addEventListener?t.addEventListener(e,n,!1):void 0)||(t["on"+e]=n)},null==T&&(T=function(t){return setTimeout(t,50)},h=function(t){return clearTimeout(t)}),j=function(e){var n=L(),r=function(){var t=L()-n;return 33<=t?(n=L(),e(t,function(){return T(r)})):setTimeout(r,33-t)};return r()},R=function(){var t=arguments[0],e=arguments[1],n=3<=arguments.length?B.call(arguments,2):[];return"function"==typeof t[e]?t[e].apply(t,n):t[e]},d=function(){for(var t,e,n,r=arguments[0],s=2<=arguments.length?B.call(arguments,1):[],o=0,i=s.length;o<i;o++)if(e=s[o])for(t in e)J.call(e,t)&&(n=e[t],null!=r[t]&&"object"==typeof r[t]&&null!=n&&"object"==typeof n?d(r[t],n):r[t]=n);return r},p=function(t){for(var e,n,r=e=0,s=0,o=t.length;s<o;s++)n=t[s],r+=Math.abs(n),e++;return r/e},m=function(t,e){var n,r;if(null==t&&(t="options"),null==e&&(e=!0),r=document.querySelector("[data-pace-"+t+"]")){if(n=r.getAttribute("data-pace-"+t),!e)return n;try{return JSON.parse(n)}catch(t){return"undefined"!=typeof console&&null!==console?console.error("Error parsing inline pace options",t):void 0}}},V.prototype.on=function(t,e,n,r){var s;return null==r&&(r=!1),null==this.bindings&&(this.bindings={}),null==(s=this.bindings)[t]&&(s[t]=[]),this.bindings[t].push({handler:e,ctx:n,once:r})},V.prototype.once=function(t,e,n){return this.on(t,e,n,!0)},V.prototype.off=function(t,e){var n,r,s;if(null!=(null!=(r=this.bindings)?r[t]:void 0)){if(null==e)return delete this.bindings[t];for(n=0,s=[];n<this.bindings[t].length;)this.bindings[t][n].handler===e?s.push(this.bindings[t].splice(n,1)):s.push(n++);return s}},V.prototype.trigger=function(){var t,e,n,r,s,o,i=arguments[0],a=2<=arguments.length?B.call(arguments,1):[];if(null!=(r=this.bindings)&&r[i]){for(n=0,o=[];n<this.bindings[i].length;)e=(s=this.bindings[i][n]).handler,t=s.ctx,s=s.once,e.apply(null!=t?t:this,a),s?o.push(this.bindings[i].splice(n,1)):o.push(n++);return o}},G=V,y=window.Pace||{},window.Pace=y,d(y,G.prototype),P=y.options=d({},f,window.paceOptions,m()),W=0,D=(I=["ajax","document","eventLag","elements"]).length;W<D;W++)!0===P[A=I[W]]&&(P[A]=f[A]);function Y(){return Y.__super__.constructor.apply(this,arguments)}function Z(){this.progress=0}function $(){this.bindings={}}function tt(){var e,o=this;tt.__super__.constructor.apply(this,arguments),e=function(r){var s=r.open;return r.open=function(t,e,n){return M(t)&&o.trigger("request",{type:t,url:e,request:r}),s.apply(r,arguments)}},window.XMLHttpRequest=function(t){t=new U(t);return e(t),t};try{g(window.XMLHttpRequest,U)}catch(t){}if(null!=F){window.XDomainRequest=function(){var t=new F;return e(t),t};try{g(window.XDomainRequest,F)}catch(t){}}if(null!=_&&P.ajax.trackWebSockets){window.WebSocket=function(t,e){var n=null!=e?new _(t,e):new _(t);return M("socket")&&o.trigger("request",{type:"socket",url:t,protocols:e,request:n}),n};try{g(window.WebSocket,_)}catch(t){}}}function et(){this.complete=o(this.complete,this);var t=this;this.elements=[],k().on("request",function(){return t.watch.apply(t,arguments)})}function nt(t){var e,n,r,s;for(null==t&&(t={}),this.complete=o(this.complete,this),this.elements=[],null==t.selectors&&(t.selectors=[]),n=0,r=(s=t.selectors).length;n<r;n++)e=s[n],this.elements.push(new i(e,this.complete))}function rt(t,e){this.selector=t,this.completeCallback=e,this.progress=0,this.check()}function st(){var t,e,n=this;this.progress=null!=(e=this.states[document.readyState])?e:100,t=document.onreadystatechange,document.onreadystatechange=function(){return null!=n.states[document.readyState]&&(n.progress=n.states[document.readyState]),"function"==typeof t?t.apply(null,arguments):void 0}}function ot(t){this.source=t,this.last=this.sinceLastUpdate=0,this.rate=P.initialRate,this.catchup=0,this.progress=this.lastProgress=0,null!=this.source&&(this.progress=R(this.source,"progress"))}G=Error,K(Y,G),n=Y,Z.prototype.getElement=function(){var t;if(null==this.el){if(!(t=document.querySelector(P.target)))throw new n;this.el=document.createElement("div"),this.el.className="pace pace-active",document.body.className=document.body.className.replace(/(pace-done )|/,"pace-running ");var e=""!==P.className?" "+P.className:"";this.el.innerHTML='<div class="pace-progress'+e+'"><div class="pace-progress-inner"></div></div><div class="pace-activity"></div>',null!=t.firstChild?t.insertBefore(this.el,t.firstChild):t.appendChild(this.el)}return this.el},Z.prototype.finish=function(){var t=this.getElement();return t.className=t.className.replace("pace-active","pace-inactive"),document.body.className=document.body.className.replace("pace-running ","pace-done ")},Z.prototype.update=function(t){return this.progress=t,y.trigger("progress",t),this.render()},Z.prototype.destroy=function(){try{this.getElement().parentNode.removeChild(this.getElement())}catch(t){n=t}return this.el=void 0},Z.prototype.render=function(){var t,e,n,r,s,o,i;if(null==document.querySelector(P.target))return!1;for(t=this.getElement(),r="translate3d("+this.progress+"%, 0, 0)",s=0,o=(i=["webkitTransform","msTransform","transform"]).length;s<o;s++)e=i[s],t.children[0].style[e]=r;return(!this.lastRenderedProgress||this.lastRenderedProgress|0!==this.progress|0)&&(t.children[0].setAttribute("data-progress-text",(0|this.progress)+"%"),100<=this.progress?n="99":(n=this.progress<10?"0":"",n+=0|this.progress),t.children[0].setAttribute("data-progress",""+n)),y.trigger("change",this.progress),this.lastRenderedProgress=this.progress},Z.prototype.done=function(){return 100<=this.progress},c=Z,$.prototype.trigger=function(t,e){var n,r,s,o,i;if(null!=this.bindings[t]){for(i=[],r=0,s=(o=this.bindings[t]).length;r<s;r++)n=o[r],i.push(n.call(this,e));return i}},$.prototype.on=function(t,e){var n;return null==(n=this.bindings)[t]&&(n[t]=[]),this.bindings[t].push(e)},m=$,U=window.XMLHttpRequest,F=window.XDomainRequest,_=window.WebSocket,g=function(t,e){var n,r=[];for(n in e.prototype)try{null==t[n]&&"function"!=typeof e[n]?"function"==typeof Object.defineProperty?r.push(Object.defineProperty(t,n,{get:function(t){return function(){return e.prototype[t]}}(n),configurable:!0,enumerable:!0})):r.push(t[n]=e.prototype[n]):r.push(void 0)}catch(t){0}return r},q=[],y.ignore=function(){var t=arguments[0],e=2<=arguments.length?B.call(arguments,1):[];return q.unshift("ignore"),e=t.apply(null,e),q.shift(),e},y.track=function(){var t=arguments[0],e=2<=arguments.length?B.call(arguments,1):[];return q.unshift("track"),e=t.apply(null,e),q.shift(),e},M=function(t){if(null==t&&(t="GET"),"track"===q[0])return"force";if(!q.length&&P.ajax){if("socket"===t&&P.ajax.trackWebSockets)return!0;if(t=t.toUpperCase(),0<=Q.call(P.ajax.trackMethods,t))return!0}return!1},K(tt,m),t=tt,X=null,E=function(t){for(var e,n=P.ajax.ignoreURLs,r=0,s=n.length;r<s;r++)if("string"==typeof(e=n[r])){if(-1!==t.indexOf(e))return!0}else if(e.test(t))return!0;return!1},(k=function(){return null==X&&(X=new t),X})().on("request",function(t){var o,i=t.type,a=t.request,e=t.url;if(!E(e))return y.running||!1===P.restartOnRequestAfter&&"force"!==M(i)?void 0:(o=arguments,"boolean"==typeof(e=P.restartOnRequestAfter||0)&&(e=0),setTimeout(function(){var t,e,n,r,s="socket"===i?a.readyState<1:0<(s=a.readyState)&&s<4;if(s){for(y.restart(),r=[],t=0,e=(n=y.sources).length;t<e;t++){if((A=n[t])instanceof u){A.watch.apply(A,o);break}r.push(void 0)}return r}},e))}),et.prototype.watch=function(t){var e=t.type,n=t.request,t=t.url;if(!E(t))return n=new("socket"===e?r:s)(n,this.complete),this.elements.push(n)},et.prototype.complete=function(e){return this.elements=this.elements.filter(function(t){return t!==e})},u=et,s=function(e,n){var t,r,s,o,i=this;if(this.progress=0,null!=window.ProgressEvent)for(a(e,"progress",function(t){return t.lengthComputable?i.progress=100*t.loaded/t.total:i.progress=i.progress+(100-i.progress)/2}),t=0,r=(o=["load","abort","timeout","error"]).length;t<r;t++)a(e,o[t],function(){return n(i),i.progress=100});else s=e.onreadystatechange,e.onreadystatechange=function(){var t;return 0===(t=e.readyState)||4===t?(n(i),i.progress=100):3===e.readyState&&(i.progress=50),"function"==typeof s?s.apply(null,arguments):void 0}},r=function(t,e){for(var n,r=this,s=this.progress=0,o=(n=["error","open"]).length;s<o;s++)a(t,n[s],function(){return e(r),r.progress=100})},nt.prototype.complete=function(e){return this.elements=this.elements.filter(function(t){return t!==e})},G=nt,rt.prototype.check=function(){var t=this;return document.querySelector(this.selector)?this.done():setTimeout(function(){return t.check()},P.elements.checkInterval)},rt.prototype.done=function(){return this.completeCallback(this),this.completeCallback=null,this.progress=100},i=rt,st.prototype.states={loading:0,interactive:50,complete:100},K=st,m=function(){var e,n,r,s,o,i=this;this.progress=0,o=[],s=0,r=L(),n=setInterval(function(){var t=L()-r-50;return r=L(),o.push(t),o.length>P.eventLag.sampleCount&&o.shift(),e=p(o),++s>=P.eventLag.minSamples&&e<P.eventLag.lagThreshold?(i.progress=100,clearInterval(n)):i.progress=3/(e+3)*100},50)},ot.prototype.tick=function(t,e){return null==e&&(e=R(this.source,"progress")),100<=e&&(this.done=!0),e===this.last?this.sinceLastUpdate+=t:(this.sinceLastUpdate&&(this.rate=(e-this.last)/this.sinceLastUpdate),this.catchup=(e-this.progress)/P.catchupTime,this.sinceLastUpdate=0,this.last=e),e>this.progress&&(this.progress+=this.catchup*t),e=1-Math.pow(this.progress/100,P.easeFactor),this.progress+=e*this.rate*t,this.progress=Math.min(this.lastProgress+P.maxProgressPerFrame,this.progress),this.progress=Math.max(0,this.progress),this.progress=Math.min(100,this.progress),this.lastProgress=this.progress,this.progress},v=ot,b=e=N=w=O=C=null,y.running=!1,S=function(){if(P.restartOnPushState)return y.restart()},null!=window.history.pushState&&(H=window.history.pushState,window.history.pushState=function(){return S(),H.apply(window.history,arguments)}),null!=window.history.replaceState&&(z=window.history.replaceState,window.history.replaceState=function(){return S(),z.apply(window.history,arguments)}),l={ajax:u,elements:G,document:K,eventLag:m},(x=function(){var t,e,n,r,s,o,i,a;for(y.sources=C=[],e=0,r=(o=["ajax","elements","document","eventLag"]).length;e<r;e++)!1!==P[t=o[e]]&&C.push(new l[t](P[t]));for(n=0,s=(a=null!=(i=P.extraSources)?i:[]).length;n<s;n++)A=a[n],C.push(new A(P));return y.bar=w=new c,O=[],N=new v})(),y.stop=function(){return y.trigger("stop"),y.running=!1,w.destroy(),b=!0,null!=e&&("function"==typeof h&&h(e),e=null),x()},y.restart=function(){return y.trigger("restart"),y.stop(),y.start()},y.go=function(){var m;return y.running=!0,w.render(),m=L(),b=!1,e=j(function(t,e){w.progress;for(var n,r,s,o,i,a,u,c,l,p,h=a=0,f=!0,d=u=0,g=C.length;u<g;d=++u)for(A=C[d],i=null!=O[d]?O[d]:O[d]=[],s=c=0,l=(r=null!=(p=A.elements)?p:[A]).length;c<l;s=++c)o=r[s],f&=(o=null!=i[s]?i[s]:i[s]=new v(o)).done,o.done||(h++,a+=o.tick(t));return n=a/h,w.update(N.tick(t,n)),w.done()||f||b?(w.update(100),y.trigger("done"),setTimeout(function(){return w.finish(),y.running=!1,y.trigger("hide")},Math.max(P.ghostTime,Math.max(P.minTime-(L()-m),0)))):e()})},y.start=function(t){d(P,t),y.running=!0;try{w.render()}catch(t){n=t}return document.querySelector(".pace")?(y.trigger("start"),y.go()):setTimeout(y.start,50)},"function"==typeof define&&define.amd?define(function(){return y}):"object"==typeof exports?module.exports=y:P.startOnPageLoad&&y.start()}).call(this);"></script>
<link href='data:application/manifest+json;charset=utf-8,%7B%0A%20%20%20%20%22name%22%3A%20%22VOTYMEHJONG%22%2C%0A%20%20%20%20%22short_name%22%3A%20%22VOTYMEHJONG%22%2C%0A%20%20%20%20%22lang%22%3A%20%22pt-BR%22%2C%0A%20%20%20%20%22description%22%3A%20%22VOTYMEHJONG%20ADAPT%C3%81VEL.%20APLICATIVO%20BY%20DEVELOPER%20DAVIDSONBPE.%22%2C%0A%20%20%20%20%22theme_color%22%3A%20%22%23222%22%2C%0A%20%20%20%20%22background_color%22%3A%20%22%23222%22%2C%0A%20%20%20%20%22display%22%3A%20%22standalone%22%2C%0A%20%20%20%20%22orientation%22%3A%20%22any%22%2C%0A%20%20%20%20%22start_url%22%3A%20%22https%3A%2F%2Fkexel-git.static.hf.space%2FVotyMehjong.html%3FInstalle%3Dvotymehjong%22%2C%0A%20%20%20%20%22id%22%3A%20%22votymehjong%22%2C%0A%20%20%20%20%22iarc_rating_id%22%3A%20%221455-UI88-NH10-NM36-KK89-JD12-PP23-GT20-2025%22%2C%0A%20%20%20%20%22dir%22%3A%20%22auto%22%2C%0A%20%20%20%20%22display_override%22%3A%20%5B%0A%20%20%20%20%20%20%20%20%22standalone%22%0A%20%20%20%20%5D%2C%0A%20%20%20%20%22categories%22%3A%20%5B%0A%20%20%20%20%20%20%20%20%22books%22%0A%20%20%20%20%5D%2C%0A%20%20%20%20%22icons%22%3A%20%5B%0A%20%20%20%20%20%20%20%20%7B%0A%20%20%20%20%20%20%20%20%20%20%20%20%22src%22%3A%20%22https%3A%2F%2Fplacehold.co%2F144%2F3b2d50%2Fbdbac1.png%3Ffont%3Droboto%26text%3DVOTY.MEHJONG%22%2C%0A%20%20%20%20%20%20%20%20%20%20%20%20%22sizes%22%3A%20%22144x144%22%2C%0A%20%20%20%20%20%20%20%20%20%20%20%20%22type%22%3A%20%22image%2Fpng%22%0A%20%20%20%20%20%20%20%20%7D%2C%0A%20%20%20%20%20%20%20%20%7B%0A%20%20%20%20%20%20%20%20%20%20%20%20%22src%22%3A%20%22https%3A%2F%2Fplacehold.co%2F36%2F3b2d50%2Fbdbac1.png%3Ffont%3Droboto%26text%3DVOTY.MEHJONG%22%2C%0A%20%20%20%20%20%20%20%20%20%20%20%20%22sizes%22%3A%20%2236x36%22%2C%0A%20%20%20%20%20%20%20%20%20%20%20%20%22type%22%3A%20%22image%2Fpng%22%0A%20%20%20%20%20%20%20%20%7D%2C%0A%20%20%20%20%20%20%20%20%7B%0A%20%20%20%20%20%20%20%20%20%20%20%20%22src%22%3A%20%22https%3A%2F%2Fplacehold.co%2F48%2F3b2d50%2Fbdbac1.png%3Ffont%3Droboto%26text%3DVOTY.MEHJONG%22%2C%0A%20%20%20%20%20%20%20%20%20%20%20%20%22sizes%22%3A%20%2248x48%22%2C%0A%20%20%20%20%20%20%20%20%20%20%20%20%22type%22%3A%20%22image%2Fpng%22%0A%20%20%20%20%20%20%20%20%7D%2C%0A%20%20%20%20%20%20%20%20%7B%0A%20%20%20%20%20%20%20%20%20%20%20%20%22src%22%3A%20%22https%3A%2F%2Fplacehold.co%2F72%2F3b2d50%2Fbdbac1.png%3Ffont%3Droboto%26text%3DVOTY.MEHJONG%22%2C%0A%20%20%20%20%20%20%20%20%20%20%20%20%22sizes%22%3A%20%2272x72%22%2C%0A%20%20%20%20%20%20%20%20%20%20%20%20%22type%22%3A%20%22image%2Fpng%22%0A%20%20%20%20%20%20%20%20%7D%2C%0A%20%20%20%20%20%20%20%20%7B%0A%20%20%20%20%20%20%20%20%20%20%20%20%22src%22%3A%20%22https%3A%2F%2Fplacehold.co%2F96%2F3b2d50%2Fbdbac1.png%3Ffont%3Droboto%26text%3DVOTY.MEHJONG%22%2C%0A%20%20%20%20%20%20%20%20%20%20%20%20%22sizes%22%3A%20%2296x96%22%2C%0A%20%20%20%20%20%20%20%20%20%20%20%20%22type%22%3A%20%22image%2Fpng%22%0A%20%20%20%20%20%20%20%20%7D%2C%0A%20%20%20%20%20%20%20%20%7B%0A%20%20%20%20%20%20%20%20%20%20%20%20%22src%22%3A%20%22https%3A%2F%2Fplacehold.co%2F192%2F3b2d50%2Fbdbac1.png%3Ffont%3Droboto%26text%3DVOTY.MEHJONG%22%2C%0A%20%20%20%20%20%20%20%20%20%20%20%20%22sizes%22%3A%20%22192x192%22%2C%0A%20%20%20%20%20%20%20%20%20%20%20%20%22type%22%3A%20%22image%2Fpng%22%2C%0A%20%20%20%20%20%20%20%20%20%20%20%20%22purpose%22%3A%20%22any%20maskable%20monochrome%22%0A%20%20%20%20%20%20%20%20%7D%2C%0A%20%20%20%20%20%20%20%20%7B%0A%20%20%20%20%20%20%20%20%20%20%20%20%22src%22%3A%20%22https%3A%2F%2Fplacehold.co%2F512%2F3b2d50%2Fbdbac1.png%3Ffont%3Droboto%26text%3DVOTY.MEHJONG%22%2C%0A%20%20%20%20%20%20%20%20%20%20%20%20%22sizes%22%3A%20%22512x512%22%2C%0A%20%20%20%20%20%20%20%20%20%20%20%20%22type%22%3A%20%22image%2Fpng%22%0A%20%20%20%20%20%20%20%20%7D%0A%20%20%20%20%5D%2C%0A%20%20%20%20%22screenshots%22%3A%20%5B%0A%20%20%20%20%20%20%20%20%7B%0A%20%20%20%20%20%20%20%20%20%20%20%20%22src%22%3A%20%22https%3A%2F%2Fplacehold.co%2F512%2F3b2d50%2Fbdbac1.png%3Ffont%3Droboto%26text%3DVOTY.MEHJONG%22%2C%0A%20%20%20%20%20%20%20%20%20%20%20%20%22sizes%22%3A%20%22512x512%22%2C%0A%20%20%20%20%20%20%20%20%20%20%20%20%22type%22%3A%20%22image%2Fpng%22%2C%0A%20%20%20%20%20%20%20%20%20%20%20%20%22platform%22%3A%20%22wide%22%0A%20%20%20%20%20%20%20%20%7D%0A%20%20%20%20%5D%0A%7D' rel='manifest'/>
<script> if (navigator.serviceWorker) { navigator.serviceWorker.register } </script>
<script src="https://cdn.jsdelivr.net/gh/davserv/d-framework@refs/heads/cdn.tailwindcss/tailwindcss.js"></script>
<link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/@fortawesome/fontawesome-free@6/css/all.min.css"/>
<style>
@keyframes tileHover {
0% { transform: translateY(0); }
50% { transform: translateY(-5px); }
100% { transform: translateY(0); }
}
.tile-hover:hover {
animation: tileHover 0.3s ease;
transform: translateY(-5px);
box-shadow: 0 10px 15px -3px rgba(0, 0, 0, 0.2);
}
.tile-selected {
transform: translateY(-15px);
box-shadow: 0 20px 25px -5px rgba(0, 0, 0, 0.2);
z-index: 10;
}
.tile-matched {
animation: fadeOut 0.5s ease forwards;
}
@keyframes fadeOut {
to {
opacity: 0;
transform: scale(0.8);
}
}
.board-container {
perspective: 1000px;
}
.tile {
transition: all 0.3s ease;
transform-style: preserve-3d;
}
.tile-inner {
position: relative;
width: 100%;
height: 100%;
transform-style: preserve-3d;
}
.tile-front, .tile-back {
position: absolute;
width: 100%;
height: 100%;
backface-visibility: hidden;
border-radius: 8px;
display: flex;
align-items: center;
justify-content: center;
}
.tile-front {
background: linear-gradient(145deg, #f0f0f0, #ffffff);
transform: rotateY(180deg);
}
.tile-back {
background: linear-gradient(145deg, #4f46e5, #7c3aed);
color: white;
}
.flipped {
transform: rotateY(180deg);
}
.level-complete {
animation: pulse 2s infinite;
}
@keyframes pulse {
0% { transform: scale(1); }
50% { transform: scale(1.05); }
100% { transform: scale(1); }
}
/* Mobile-specific styles */
@media (max-width: 640px) {
.tile-front span, .tile-back i {
font-size: 50px !important;
}
.header-container {
flex-direction: column;
align-items: center;
gap: 1rem;
}
.stats-container {
width: 100%;
justify-content: space-between;
}
.controls-container {
flex-direction: column;
gap: 0.5rem;
}
.controls-container > div {
width: 100%;
}
.controls-container button {
width: 100%;
padding: 0.5rem;
}
.modal-content {
width: 90%;
padding: 1.5rem;
}
}
.text-mtg{
font-size: 50px !important;
}
</style>
</head>
<body class="bg-gray-100 min-h-screen font-sans">
<div class="container mx-auto px-2 sm:px-4 py-4 sm:py-8">
<!-- Header - Made responsive with flex column on mobile -->
<header class="header-container flex flex-col sm:flex-row justify-between items-center mb-4 sm:mb-8">
<div class="flex items-center mb-2 sm:mb-0">
<i class="fa-brands fa-fort-awesome text-3xl sm:text-4xl text-purple-600 mr-2 sm:mr-3"></i>
<h1 class="text-2xl sm:text-3xl font-bold text-gray-800 uppercase">Voty Mehjong</h1>
</div>
<div class="stats-container flex items-center space-x-2 sm:space-x-4 w-full sm:w-auto justify-center sm:justify-start">
<div class="bg-white rounded-lg shadow p-2 sm:p-3 flex items-center text-sm sm:text-base">
<i class="fas fa-clock text-purple-600 mr-1 sm:mr-2 text-sm sm:text-base"></i>
<span id="timer" class="font-bold">00:00</span>
</div>
<div class="bg-white rounded-lg shadow p-2 sm:p-3 flex items-center text-sm sm:text-base">
<i class="fas fa-layer-group text-purple-600 mr-1 sm:mr-2 text-sm sm:text-base"></i>
<span id="level" class="font-bold">Level 1</span>
</div>
<div class="bg-white rounded-lg shadow p-2 sm:p-3 flex items-center text-sm sm:text-base">
<i class="fas fa-star text-yellow-500 mr-1 sm:mr-2 text-sm sm:text-base"></i>
<span id="score" class="font-bold">0</span>
</div>
</div>
</header>
<!-- Game Controls - Stacked vertically on mobile -->
<div class="controls-container flex flex-col sm:flex-row justify-between mb-4 sm:mb-6">
<div class="flex space-x-2 sm:space-x-3 mb-2 sm:mb-0">
<button id="new-game" class="bg-purple-600 hover:bg-purple-700 text-white px-3 sm:px-4 py-1 sm:py-2 rounded-lg shadow flex items-center text-sm sm:text-base">
<i class="fas fa-plus-circle mr-1 sm:mr-2"></i> New Game
</button>
<button id="hint" class="bg-amber-500 hover:bg-amber-600 text-white px-3 sm:px-4 py-1 sm:py-2 rounded-lg shadow flex items-center text-sm sm:text-base">
<i class="fas fa-lightbulb mr-1 sm:mr-2"></i> Hint
</button>
</div>
<div>
<button id="sound-toggle" class="bg-gray-200 hover:bg-gray-300 text-gray-800 px-3 sm:px-4 py-1 sm:py-2 rounded-lg shadow flex items-center text-sm sm:text-base">
<i class="fas fa-volume-up mr-1 sm:mr-2"></i> Sound On
</button>
</div>
</div>
<!-- Game Board - Responsive grid with smaller tiles on mobile -->
<div class="board-container bg-white rounded-xl shadow-xl p-3 sm:p-6 mb-4 sm:mb-6 overflow-auto">
<div id="board" class="grid grid-cols-4 sm:grid-cols-6 md:grid-cols-8 gap-2 sm:gap-3 mx-auto"></div>
</div>
<!-- Game Status -->
<div id="game-status" class="text-center mb-4 sm:mb-6 hidden">
<div class="inline-block bg-green-100 border-l-4 border-green-500 text-green-700 p-3 sm:p-4 rounded-lg text-sm sm:text-base">
<p class="font-bold">Level Complete!</p>
</div>
</div>
<!-- Level Complete Modal - Responsive sizing -->
<div id="level-complete-modal" class="fixed inset-0 bg-black bg-opacity-50 flex items-center justify-center hidden z-50 p-2 sm:p-4">
<div class="modal-content bg-white rounded-xl shadow-2xl p-4 sm:p-8 w-full max-w-sm sm:max-w-md level-complete">
<div class="text-center">
<div class="text-4xl sm:text-6xl text-yellow-500 mb-2 sm:mb-4">
<i class="fas fa-trophy"></i>
</div>
<h2 class="text-xl sm:text-3xl font-bold text-gray-800 mb-1 sm:mb-2">Level Complete!</h2>
<p class="text-sm sm:text-base text-gray-600 mb-4 sm:mb-6">Great job! Ready for the next challenge?</p>
<div class="grid grid-cols-2 gap-2 sm:gap-4 mb-4 sm:mb-6">
<div class="bg-purple-50 rounded-lg p-2 sm:p-3">
<p class="text-xs sm:text-sm text-purple-600">Time</p>
<p id="level-time" class="font-bold text-lg sm:text-xl">00:45</p>
</div>
<div class="bg-purple-50 rounded-lg p-2 sm:p-3">
<p class="text-xs sm:text-sm text-purple-600">Score</p>
<p id="level-score" class="font-bold text-lg sm:text-xl">+250</p>
</div>
</div>
<button id="next-level" class="w-full bg-purple-600 hover:bg-purple-700 text-white py-2 sm:py-3 rounded-lg shadow-lg font-bold text-sm sm:text-base">
Next Level <i class="fas fa-arrow-right ml-1 sm:ml-2"></i>
</button>
</div>
</div>
</div>
<!-- Game Over Modal - Responsive sizing -->
<div id="game-over-modal" class="fixed inset-0 bg-black bg-opacity-50 flex items-center justify-center hidden z-50 p-2 sm:p-4">
<div class="modal-content bg-white rounded-xl shadow-2xl p-4 sm:p-8 w-full max-w-sm sm:max-w-md">
<div class="text-center">
<div class="text-4xl sm:text-6xl text-red-500 mb-2 sm:mb-4">
<i class="fas fa-gamepad"></i>
</div>
<h2 class="text-xl sm:text-3xl font-bold text-gray-800 mb-1 sm:mb-2">Game Over</h2>
<p class="text-sm sm:text-base text-gray-600 mb-4 sm:mb-6">Better luck next time!</p>
<div class="grid grid-cols-2 gap-2 sm:gap-4 mb-4 sm:mb-6">
<div class="bg-purple-50 rounded-lg p-2 sm:p-3">
<p class="text-xs sm:text-sm text-purple-600">Level Reached</p>
<p id="final-level" class="font-bold text-lg sm:text-xl">3</p>
</div>
<div class="bg-purple-50 rounded-lg p-2 sm:p-3">
<p class="text-xs sm:text-sm text-purple-600">Total Score</p>
<p id="final-score" class="font-bold text-lg sm:text-xl">750</p>
</div>
</div>
<button id="play-again" class="w-full bg-purple-600 hover:bg-purple-700 text-white py-2 sm:py-3 rounded-lg shadow-lg font-bold text-sm sm:text-base">
Play Again <i class="fas fa-redo ml-1 sm:ml-2"></i>
</button>
</div>
</div>
</div>
</div>
<!-- Audio elements for sound effects -->
<audio id="flip-sound" src="https://kexel-git.static.hf.space/gsom/zapsplat_multimedia_button_click.mp3" preload="auto"></audio>
<audio id="match-sound" src="https://kexel-git.static.hf.space/gsom/positive-beeps.mp3" preload="auto"></audio>
<audio id="mismatch-sound" src="https://kexel-git.static.hf.space/gsom/mixkit-quick-jump-arcade-game-239.mp3" preload="auto"></audio>
<audio id="hint-sound" src="https://kexel-git.static.hf.space/gsom/unlock-game-notification.mp3" preload="auto"></audio>
<audio id="level-complete-sound" src="https://kexel-git.static.hf.space/gsom/2015-preview.mp3" preload="auto"></audio>
<audio id="game-over-sound" src="https://kexel-git.static.hf.space/gsom/2027-preview.mp3" preload="auto"></audio>
<audio id="button-click-sound" src="https://kexel-git.static.hf.space/gsom/mixkit-mouse-click-close-1113.mp3" preload="auto"></audio>
<script>
document.addEventListener('DOMContentLoaded', () => {
// Game state
const state = {
board: [],
level: 1,
score: 0,
time: 0,
timerInterval: null,
selectedTiles: [],
matchedPairs: 0,
totalPairs: 0,
soundEnabled: true,
gameActive: false,
isMobile: window.innerWidth < 640 // Check if mobile device
};
// Audio elements
const flipSound = document.getElementById('flip-sound');
const matchSound = document.getElementById('match-sound');
const mismatchSound = document.getElementById('mismatch-sound');
const hintSound = document.getElementById('hint-sound');
const levelCompleteSound = document.getElementById('level-complete-sound');
const gameOverSound = document.getElementById('game-over-sound');
const buttonClickSound = document.getElementById('button-click-sound');
// DOM elements
const boardElement = document.getElementById('board');
const timerElement = document.getElementById('timer');
const levelElement = document.getElementById('level');
const scoreElement = document.getElementById('score');
const newGameButton = document.getElementById('new-game');
const hintButton = document.getElementById('hint');
const soundToggleButton = document.getElementById('sound-toggle');
const gameStatusElement = document.getElementById('game-status');
const levelCompleteModal = document.getElementById('level-complete-modal');
const gameOverModal = document.getElementById('game-over-modal');
const nextLevelButton = document.getElementById('next-level');
const playAgainButton = document.getElementById('play-again');
const levelTimeElement = document.getElementById('level-time');
const levelScoreElement = document.getElementById('level-score');
const finalLevelElement = document.getElementById('final-level');
const finalScoreElement = document.getElementById('final-score');
// Tile types (Vita Mahjong style)
const tileTypes = [
'1m', '2m', '3m', '4m', '5m', '6m', '7m', '8m', '9m', // Characters
'1s', '2s', '3s', '4s', '5s', '6s', '7s', '8s', '9s', // Bamboo
'1p', '2p', '3p', '4p', '5p', '6p', '7p', '8p', '9p', // Circles
'ew', 'sw', 'ww', 'nw', // Winds
'wd', 'gd', 'rd' // Dragons
];
// Initialize game
function initGame() {
state.level = 1;
state.score = 0;
state.time = 0;
state.matchedPairs = 0;
state.gameActive = true;
clearInterval(state.timerInterval);
startTimer();
updateUI();
createBoard();
// Play button click sound
playSound('button-click');
}
// Create game board based on current level
function createBoard() {
boardElement.innerHTML = '';
state.board = [];
state.selectedTiles = [];
state.matchedPairs = 0;
// Determine number of pairs based on level and screen size
const basePairs = state.isMobile ? 2 : 4;
const pairs = Math.min(basePairs + state.level, state.isMobile ? 12 : 32);
state.totalPairs = pairs;
// Create array of tile pairs
let tiles = [];
const availableTypes = [...tileTypes].sort(() => 0.5 - Math.random()).slice(0, pairs);
availableTypes.forEach(type => {
tiles.push(type, type);
});
// Shuffle tiles
tiles = shuffleArray(tiles);
// Determine grid columns based on screen size
const cols = state.isMobile ? 4 : (window.innerWidth < 768 ? 6 : 8);
const rows = Math.ceil(tiles.length / cols);
for (let i = 0; i < rows; i++) {
const row = [];
for (let j = 0; j < cols; j++) {
const index = i * cols + j;
if (index < tiles.length) {
row.push({
type: tiles[index],
flipped: false,
matched: false,
row: i,
col: j
});
} else {
row.push(null); // Empty slot for uneven boards
}
}
state.board.push(row);
}
// Render tiles
renderBoard();
}
// Render the game board
function renderBoard() {
boardElement.innerHTML = '';
// Calculate grid columns based on screen size
const cols = state.isMobile ? 4 : (window.innerWidth < 768 ? 6 : 8);
boardElement.className = `grid gap-2 sm:gap-3 mx-auto`;
boardElement.style.gridTemplateColumns = `repeat(${cols}, minmax(0, 1fr))`;
state.board.forEach((row, rowIndex) => {
row.forEach((tile, colIndex) => {
if (!tile) return;
const tileElement = document.createElement('div');
tileElement.className = `tile aspect-square cursor-pointer transition-all duration-300 ${tile.matched ? 'opacity-0' : ''}`;
tileElement.innerHTML = `
<div class="border-t-2 border-r-2 border-indigo-500/50 border-l-8 border-b-8 tile-inner ${
tile.flipped ? "flipped" : ""
}">
<div class="rounded-none tile-back flex items-center justify-center">
<i class="fa-brands fa-fort-awesome text-sm sm:text-5xl"></i>
</div>
<div class="tile-front">
<span class="text-xl sm:text-4xl font-bold">${getTileSymbol(
tile.type
)}</span>
</div>
</div>
`;
tileElement.addEventListener('click', () => handleTileClick(tile));
if (!tile.matched) {
tileElement.classList.add('tile-hover');
}
boardElement.appendChild(tileElement);
});
});
}
// Handle tile click
function handleTileClick(tile) {
if (!state.gameActive || tile.matched || tile.flipped || state.selectedTiles.length >= 2) {
return;
}
// Flip the tile
tile.flipped = true;
state.selectedTiles.push(tile);
// Play sound
playSound('flip');
renderBoard();
// Check for match if two tiles are selected
if (state.selectedTiles.length === 2) {
const [tile1, tile2] = state.selectedTiles;
if (tile1.type === tile2.type) {
// Match found
tile1.matched = true;
tile2.matched = true;
state.matchedPairs++;
state.score += 50 * state.level;
// Play success sound
playSound('match');
// Check if level is complete
if (state.matchedPairs === state.totalPairs) {
levelComplete();
}
// Clear selection after delay
setTimeout(() => {
state.selectedTiles = [];
renderBoard();
}, 500);
} else {
// No match
setTimeout(() => {
tile1.flipped = false;
tile2.flipped = false;
state.selectedTiles = [];
renderBoard();
// Play mismatch sound
playSound('mismatch');
}, 1000);
}
}
updateUI();
}
// Level complete
function levelComplete() {
state.gameActive = false;
clearInterval(state.timerInterval);
// Calculate bonus points based on time
const timeBonus = Math.max(0, 300 - state.time);
state.score += timeBonus;
// Show level complete modal
levelTimeElement.textContent = formatTime(state.time);
levelScoreElement.textContent = `+${timeBonus}`;
levelCompleteModal.classList.remove('hidden');
// Play level complete sound
playSound('level-complete');
updateUI();
}
// Next level
function nextLevel() {
playSound('button-click');
state.level++;
state.time = 0;
state.gameActive = true;
levelCompleteModal.classList.add('hidden');
startTimer();
createBoard();
updateUI();
// Show level up message
gameStatusElement.classList.remove('hidden');
gameStatusElement.querySelector('p').textContent = `Level ${state.level}!`;
setTimeout(() => {
gameStatusElement.classList.add('hidden');
}, 2000);
}
// Game over
function gameOver() {
state.gameActive = false;
clearInterval(state.timerInterval);
// Update final stats
finalLevelElement.textContent = state.level;
finalScoreElement.textContent = state.score;
// Show game over modal
gameOverModal.classList.remove('hidden');
// Play game over sound
playSound('game-over');
}
// Start timer
function startTimer() {
state.time = 0;
updateTimerDisplay();
state.timerInterval = setInterval(() => {
state.time++;
updateTimerDisplay();
// Game over if time exceeds limit (5 minutes)
if (state.time >= 300) {
gameOver();
}
}, 1000);
}
// Update timer display
function updateTimerDisplay() {
timerElement.textContent = formatTime(state.time);
}
// Format time as MM:SS
function formatTime(seconds) {
const mins = Math.floor(seconds / 60).toString().padStart(2, '0');
const secs = (seconds % 60).toString().padStart(2, '0');
return `${mins}:${secs}`;
}
// Update UI elements
function updateUI() {
levelElement.textContent = `Level ${state.level}`;
scoreElement.textContent = state.score;
}
// Get tile symbol for display
function getTileSymbol(type) {
const symbols = {
'1m': '一', '2m': '二', '3m': '三', '4m': '四', '5m': '五',
'6m': '六', '7m': '七', '8m': '八', '9m': '九',
'1s': '1', '2s': '2', '3s': '3', '4s': '4', '5s': '5',
'6s': '6', '7s': '7', '8s': '8', '9s': '9',
'1p': '❶', '2p': '❷', '3p': '❸', '4p': '❹', '5p': '❺',
'6p': '❻', '7p': '❼', '8p': '❽', '9p': '❾',
'ew': '東', 'sw': '南', 'ww': '西', 'nw': '北',
'wd': '白', 'gd': '發', 'rd': '中'
};
return symbols[type] || type;
}
// Shuffle array
function shuffleArray(array) {
const newArray = [...array];
for (let i = newArray.length - 1; i > 0; i--) {
const j = Math.floor(Math.random() * (i + 1));
[newArray[i], newArray[j]] = [newArray[j], newArray[i]];
}
return newArray;
}
// Play sound
function playSound(type) {
if (!state.soundEnabled) return;
try {
switch (type) {
case 'flip':
flipSound.currentTime = 0;
flipSound.play();
break;
case 'match':
matchSound.currentTime = 0;
matchSound.play();
break;
case 'mismatch':
mismatchSound.currentTime = 0;
mismatchSound.play();
break;
case 'hint':
hintSound.currentTime = 0;
hintSound.play();
break;
case 'level-complete':
levelCompleteSound.currentTime = 0;
levelCompleteSound.play();
break;
case 'game-over':
gameOverSound.currentTime = 0;
gameOverSound.play();
break;
case 'button-click':
buttonClickSound.currentTime = 0;
buttonClickSound.play();
break;
}
} catch (e) {
console.error('Error playing sound:', e);
}
}
// Provide hint
function provideHint() {
if (!state.gameActive || state.matchedPairs === state.totalPairs) {
return;
}
// Play button click sound
playSound('button-click');
// Find all unflipped, unmatched tiles
const unflippedTiles = [];
state.board.forEach(row => {
row.forEach(tile => {
if (tile && !tile.flipped && !tile.matched) {
unflippedTiles.push(tile);
}
});
});
if (unflippedTiles.length < 2) return;
// Find a matching pair
const tileCount = {};
let hintTile1 = null;
let hintTile2 = null;
for (const tile of unflippedTiles) {
if (tileCount[tile.type]) {
hintTile1 = tileCount[tile.type];
hintTile2 = tile;
break;
}
tileCount[tile.type] = tile;
}
if (hintTile1 && hintTile2) {
// Highlight the hint tiles
const tileElements = boardElement.querySelectorAll('.tile');
tileElements.forEach((element, index) => {
const cols = state.isMobile ? 4 : (window.innerWidth < 768 ? 6 : 8);
const row = Math.floor(index / cols);
const col = index % cols;
const tile = state.board[row]?.[col];
if (tile === hintTile1 || tile === hintTile2) {
element.classList.add('ring-4', 'ring-yellow-400', 'ring-opacity-75');
// Remove highlight after delay
setTimeout(() => {
element.classList.remove('ring-4', 'ring-yellow-400', 'ring-opacity-75');
}, 2000);
}
});
// Deduct points for using hint
state.score = Math.max(0, state.score - 25);
updateUI();
// Play hint sound
playSound('hint');
}
}
// Event listeners
newGameButton.addEventListener('click', initGame);
hintButton.addEventListener('click', provideHint);
soundToggleButton.addEventListener('click', () => {
state.soundEnabled = !state.soundEnabled;
soundToggleButton.innerHTML = state.soundEnabled
? '<i class="fas fa-volume-up mr-1 sm:mr-2"></i> Sound On'
: '<i class="fas fa-volume-mute mr-1 sm:mr-2"></i> Sound Off';
playSound('button-click');
});
nextLevelButton.addEventListener('click', nextLevel);
playAgainButton.addEventListener('click', () => {
playSound('button-click');
gameOverModal.classList.add('hidden');
initGame();
});
// Handle window resize
window.addEventListener('resize', () => {
state.isMobile = window.innerWidth < 640;
if (state.gameActive) {
createBoard();
}
});
// Initialize the game
initGame();
});
</script>
</body>
</html>
<!--<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>Vita Mahjong</title>
<script src="https://cdn.tailwindcss.com"></script>
<link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/font-awesome/6.4.0/css/all.min.css">
<style>
@keyframes tileHover {
0% { transform: translateY(0); }
50% { transform: translateY(-5px); }
100% { transform: translateY(0); }
}
.tile-hover:hover {
animation: tileHover 0.3s ease;
transform: translateY(-5px);
box-shadow: 0 10px 15px -3px rgba(0, 0, 0, 0.2);
}
.tile-selected {
transform: translateY(-15px);
box-shadow: 0 20px 25px -5px rgba(0, 0, 0, 0.2);
z-index: 10;
}
.tile-matched {
animation: fadeOut 0.5s ease forwards;
}
@keyframes fadeOut {
to {
opacity: 0;
transform: scale(0.8);
}
}
.board-container {
perspective: 1000px;
}
.tile {
transition: all 0.3s ease;
transform-style: preserve-3d;
}
.tile-inner {
position: relative;
width: 100%;
height: 100%;
transform-style: preserve-3d;
}
.tile-front, .tile-back {
position: absolute;
width: 100%;
height: 100%;
backface-visibility: hidden;
border-radius: 8px;
display: flex;
align-items: center;
justify-content: center;
}
.tile-front {
background: linear-gradient(145deg, #f0f0f0, #ffffff);
transform: rotateY(180deg);
}
.tile-back {
background: linear-gradient(145deg, #4f46e5, #7c3aed);
color: white;
}
.flipped {
transform: rotateY(180deg);
}
.level-complete {
animation: pulse 2s infinite;
}
@keyframes pulse {
0% { transform: scale(1); }
50% { transform: scale(1.05); }
100% { transform: scale(1); }
}
</style>
</head>
<body class="bg-gray-100 min-h-screen font-sans">
<div class="container mx-auto px-4 py-8">
<header class="flex justify-between items-center mb-8">
<div class="flex items-center">
<i class="fas fa-dragon text-4xl text-purple-600 mr-3"></i>
<h1 class="text-3xl font-bold text-gray-800">Vita Mahjong</h1>
</div>
<div class="flex items-center space-x-4">
<div class="bg-white rounded-lg shadow p-3 flex items-center">
<i class="fas fa-clock text-purple-600 mr-2"></i>
<span id="timer" class="font-bold">00:00</span>
</div>
<div class="bg-white rounded-lg shadow p-3 flex items-center">
<i class="fas fa-layer-group text-purple-600 mr-2"></i>
<span id="level" class="font-bold">Level 1</span>
</div>
<div class="bg-white rounded-lg shadow p-3 flex items-center">
<i class="fas fa-star text-yellow-500 mr-2"></i>
<span id="score" class="font-bold">0</span>
</div>
</div>
</header>
<div class="flex justify-between mb-6">
<div class="flex space-x-3">
<button id="new-game" class="bg-purple-600 hover:bg-purple-700 text-white px-4 py-2 rounded-lg shadow flex items-center">
<i class="fas fa-plus-circle mr-2"></i> New Game
</button>
<button id="hint" class="bg-amber-500 hover:bg-amber-600 text-white px-4 py-2 rounded-lg shadow flex items-center">
<i class="fas fa-lightbulb mr-2"></i> Hint
</button>
</div>
<div>
<button id="sound-toggle" class="bg-gray-200 hover:bg-gray-300 text-gray-800 px-4 py-2 rounded-lg shadow flex items-center">
<i class="fas fa-volume-up mr-2"></i> Sound On
</button>
</div>
</div>
<div class="board-container bg-white rounded-xl shadow-xl p-6 mb-6">
<div id="board" class="grid grid-cols-8 gap-3 mx-auto"></div>
</div>
<div id="game-status" class="text-center mb-6 hidden">
<div class="inline-block bg-green-100 border-l-4 border-green-500 text-green-700 p-4 rounded-lg">
<p class="font-bold">Level Complete!</p>
</div>
</div>
<div id="level-complete-modal" class="fixed inset-0 bg-black bg-opacity-50 flex items-center justify-center hidden z-50">
<div class="bg-white rounded-xl shadow-2xl p-8 max-w-md w-full level-complete">
<div class="text-center">
<div class="text-6xl text-yellow-500 mb-4">
<i class="fas fa-trophy"></i>
</div>
<h2 class="text-3xl font-bold text-gray-800 mb-2">Level Complete!</h2>
<p class="text-gray-600 mb-6">Great job! Ready for the next challenge?</p>
<div class="grid grid-cols-2 gap-4 mb-6">
<div class="bg-purple-50 rounded-lg p-3">
<p class="text-sm text-purple-600">Time</p>
<p id="level-time" class="font-bold text-xl">00:45</p>
</div>
<div class="bg-purple-50 rounded-lg p-3">
<p class="text-sm text-purple-600">Score</p>
<p id="level-score" class="font-bold text-xl">+250</p>
</div>
</div>
<button id="next-level" class="w-full bg-purple-600 hover:bg-purple-700 text-white py-3 rounded-lg shadow-lg font-bold">
Next Level <i class="fas fa-arrow-right ml-2"></i>
</button>
</div>
</div>
</div>
<div id="game-over-modal" class="fixed inset-0 bg-black bg-opacity-50 flex items-center justify-center hidden z-50">
<div class="bg-white rounded-xl shadow-2xl p-8 max-w-md w-full">
<div class="text-center">
<div class="text-6xl text-red-500 mb-4">
<i class="fas fa-gamepad"></i>
</div>
<h2 class="text-3xl font-bold text-gray-800 mb-2">Game Over</h2>
<p class="text-gray-600 mb-6">Better luck next time!</p>
<div class="grid grid-cols-2 gap-4 mb-6">
<div class="bg-purple-50 rounded-lg p-3">
<p class="text-sm text-purple-600">Level Reached</p>
<p id="final-level" class="font-bold text-xl">3</p>
</div>
<div class="bg-purple-50 rounded-lg p-3">
<p class="text-sm text-purple-600">Total Score</p>
<p id="final-score" class="font-bold text-xl">750</p>
</div>
</div>
<button id="play-again" class="w-full bg-purple-600 hover:bg-purple-700 text-white py-3 rounded-lg shadow-lg font-bold">
Play Again <i class="fas fa-redo ml-2"></i>
</button>
</div>
</div>
</div>
</div>
<audio id="flip-sound" src="https://kexel-git.static.hf.space/gsom/zapsplat_multimedia_button_click.mp3" preload="auto"></audio>
<audio id="match-sound" src="https://kexel-git.static.hf.space/gsom/positive-beeps.mp3" preload="auto"></audio>
<audio id="mismatch-sound" src="https://kexel-git.static.hf.space/gsom/mixkit-quick-jump-arcade-game-239.mp3" preload="auto"></audio>
<audio id="hint-sound" src="https://kexel-git.static.hf.space/gsom/unlock-game-notification.mp3" preload="auto"></audio>
<audio id="level-complete-sound" src="https://kexel-git.static.hf.space/gsom/2015-preview.mp3" preload="auto"></audio>
<audio id="game-over-sound" src="https://kexel-git.static.hf.space/gsom/2027-preview.mp3" preload="auto"></audio>
<audio id="button-click-sound" src="https://kexel-git.static.hf.space/gsom/mixkit-mouse-click-close-1113.mp3" preload="auto"></audio>
<script>
document.addEventListener('DOMContentLoaded', () => {
// Game state
const state = {
board: [],
level: 1,
score: 0,
time: 0,
timerInterval: null,
selectedTiles: [],
matchedPairs: 0,
totalPairs: 0,
soundEnabled: true,
gameActive: false
};
// Audio elements
const flipSound = document.getElementById('flip-sound');
const matchSound = document.getElementById('match-sound');
const mismatchSound = document.getElementById('mismatch-sound');
const hintSound = document.getElementById('hint-sound');
const levelCompleteSound = document.getElementById('level-complete-sound');
const gameOverSound = document.getElementById('game-over-sound');
const buttonClickSound = document.getElementById('button-click-sound');
// DOM elements
const boardElement = document.getElementById('board');
const timerElement = document.getElementById('timer');
const levelElement = document.getElementById('level');
const scoreElement = document.getElementById('score');
const newGameButton = document.getElementById('new-game');
const hintButton = document.getElementById('hint');
const soundToggleButton = document.getElementById('sound-toggle');
const gameStatusElement = document.getElementById('game-status');
const levelCompleteModal = document.getElementById('level-complete-modal');
const gameOverModal = document.getElementById('game-over-modal');
const nextLevelButton = document.getElementById('next-level');
const playAgainButton = document.getElementById('play-again');
const levelTimeElement = document.getElementById('level-time');
const levelScoreElement = document.getElementById('level-score');
const finalLevelElement = document.getElementById('final-level');
const finalScoreElement = document.getElementById('final-score');
// Tile types (Vita Mahjong style)
const tileTypes = [
'1m', '2m', '3m', '4m', '5m', '6m', '7m', '8m', '9m', // Characters
'1s', '2s', '3s', '4s', '5s', '6s', '7s', '8s', '9s', // Bamboo
'1p', '2p', '3p', '4p', '5p', '6p', '7p', '8p', '9p', // Circles
'ew', 'sw', 'ww', 'nw', // Winds
'wd', 'gd', 'rd' // Dragons
];
// Initialize game
function initGame() {
state.level = 1;
state.score = 0;
state.time = 0;
state.matchedPairs = 0;
state.gameActive = true;
clearInterval(state.timerInterval);
startTimer();
updateUI();
createBoard();
// Play button click sound
playSound('button-click');
}
// Create game board based on current level
function createBoard() {
boardElement.innerHTML = '';
state.board = [];
state.selectedTiles = [];
state.matchedPairs = 0;
// Determine number of pairs based on level
const pairs = Math.min(4 + state.level, 32); // Max 32 pairs (64 tiles)
state.totalPairs = pairs;
// Create array of tile pairs
let tiles = [];
const availableTypes = [...tileTypes].sort(() => 0.5 - Math.random()).slice(0, pairs);
availableTypes.forEach(type => {
tiles.push(type, type);
});
// Shuffle tiles
tiles = shuffleArray(tiles);
// Create board matrix (8 columns, rows as needed)
const cols = 8;
const rows = Math.ceil(tiles.length / cols);
for (let i = 0; i < rows; i++) {
const row = [];
for (let j = 0; j < cols; j++) {
const index = i * cols + j;
if (index < tiles.length) {
row.push({
type: tiles[index],
flipped: false,
matched: false,
row: i,
col: j
});
} else {
row.push(null); // Empty slot for uneven boards
}
}
state.board.push(row);
}
// Render tiles
renderBoard();
}
// Render the game board
function renderBoard() {
boardElement.innerHTML = '';
// Calculate grid rows based on board rows
boardElement.className = `grid gap-3 mx-auto`;
boardElement.style.gridTemplateColumns = `repeat(8, minmax(0, 1fr))`;
state.board.forEach((row, rowIndex) => {
row.forEach((tile, colIndex) => {
if (!tile) return;
const tileElement = document.createElement('div');
tileElement.className = `tile aspect-square cursor-pointer transition-all duration-300 ${tile.matched ? 'opacity-0' : ''}`;
tileElement.innerHTML = `
<div class="tile-inner ${tile.flipped ? 'flipped' : ''}">
<div class="tile-back flex items-center justify-center">
<i class="fas fa-dragon text-2xl"></i>
</div>
<div class="tile-front">
<span class="text-3xl font-bold">${getTileSymbol(tile.type)}</span>
</div>
</div>
`;
tileElement.addEventListener('click', () => handleTileClick(tile));
if (!tile.matched) {
tileElement.classList.add('tile-hover');
}
boardElement.appendChild(tileElement);
});
});
}
// Handle tile click
function handleTileClick(tile) {
if (!state.gameActive || tile.matched || tile.flipped || state.selectedTiles.length >= 2) {
return;
}
// Flip the tile
tile.flipped = true;
state.selectedTiles.push(tile);
// Play sound
playSound('flip');
renderBoard();
// Check for match if two tiles are selected
if (state.selectedTiles.length === 2) {
const [tile1, tile2] = state.selectedTiles;
if (tile1.type === tile2.type) {
// Match found
tile1.matched = true;
tile2.matched = true;
state.matchedPairs++;
state.score += 50 * state.level;
// Play success sound
playSound('match');
// Check if level is complete
if (state.matchedPairs === state.totalPairs) {
levelComplete();
}
// Clear selection after delay
setTimeout(() => {
state.selectedTiles = [];
renderBoard();
}, 500);
} else {
// No match
setTimeout(() => {
tile1.flipped = false;
tile2.flipped = false;
state.selectedTiles = [];
renderBoard();
// Play mismatch sound
playSound('mismatch');
}, 1000);
}
}
updateUI();
}
// Level complete
function levelComplete() {
state.gameActive = false;
clearInterval(state.timerInterval);
// Calculate bonus points based on time
const timeBonus = Math.max(0, 300 - state.time);
state.score += timeBonus;
// Show level complete modal
levelTimeElement.textContent = formatTime(state.time);
levelScoreElement.textContent = `+${timeBonus}`;
levelCompleteModal.classList.remove('hidden');
// Play level complete sound
playSound('level-complete');
updateUI();
}
// Next level
function nextLevel() {
playSound('button-click');
state.level++;
state.time = 0;
state.gameActive = true;
levelCompleteModal.classList.add('hidden');
startTimer();
createBoard();
updateUI();
// Show level up message
gameStatusElement.classList.remove('hidden');
gameStatusElement.querySelector('p').textContent = `Level ${state.level}!`;
setTimeout(() => {
gameStatusElement.classList.add('hidden');
}, 2000);
}
// Game over
function gameOver() {
state.gameActive = false;
clearInterval(state.timerInterval);
// Update final stats
finalLevelElement.textContent = state.level;
finalScoreElement.textContent = state.score;
// Show game over modal
gameOverModal.classList.remove('hidden');
// Play game over sound
playSound('game-over');
}
// Start timer
function startTimer() {
state.time = 0;
updateTimerDisplay();
state.timerInterval = setInterval(() => {
state.time++;
updateTimerDisplay();
// Game over if time exceeds limit (5 minutes)
if (state.time >= 300) {
gameOver();
}
}, 1000);
}
// Update timer display
function updateTimerDisplay() {
timerElement.textContent = formatTime(state.time);
}
// Format time as MM:SS
function formatTime(seconds) {
const mins = Math.floor(seconds / 60).toString().padStart(2, '0');
const secs = (seconds % 60).toString().padStart(2, '0');
return `${mins}:${secs}`;
}
// Update UI elements
function updateUI() {
levelElement.textContent = `Level ${state.level}`;
scoreElement.textContent = state.score;
}
// Get tile symbol for display
function getTileSymbol(type) {
const symbols = {
'1m': '一', '2m': '二', '3m': '三', '4m': '四', '5m': '五',
'6m': '六', '7m': '七', '8m': '八', '9m': '九',
'1s': '1', '2s': '2', '3s': '3', '4s': '4', '5s': '5',
'6s': '6', '7s': '7', '8s': '8', '9s': '9',
'1p': '❶', '2p': '❷', '3p': '❸', '4p': '❹', '5p': '❺',
'6p': '❻', '7p': '❼', '8p': '❽', '9p': '❾',
'ew': '東', 'sw': '南', 'ww': '西', 'nw': '北',
'wd': '白', 'gd': '發', 'rd': '中'
};
return symbols[type] || type;
}
// Shuffle array
function shuffleArray(array) {
const newArray = [...array];
for (let i = newArray.length - 1; i > 0; i--) {
const j = Math.floor(Math.random() * (i + 1));
[newArray[i], newArray[j]] = [newArray[j], newArray[i]];
}
return newArray;
}
// Play sound
function playSound(type) {
if (!state.soundEnabled) return;
try {
switch (type) {
case 'flip':
flipSound.currentTime = 0;
flipSound.play();
break;
case 'match':
matchSound.currentTime = 0;
matchSound.play();
break;
case 'mismatch':
mismatchSound.currentTime = 0;
mismatchSound.play();
break;
case 'hint':
hintSound.currentTime = 0;
hintSound.play();
break;
case 'level-complete':
levelCompleteSound.currentTime = 0;
levelCompleteSound.play();
break;
case 'game-over':
gameOverSound.currentTime = 0;
gameOverSound.play();
break;
case 'button-click':
buttonClickSound.currentTime = 0;
buttonClickSound.play();
break;
}
} catch (e) {
console.error('Error playing sound:', e);
}
}
// Provide hint
function provideHint() {
if (!state.gameActive || state.matchedPairs === state.totalPairs) {
return;
}
// Play button click sound
playSound('button-click');
// Find all unflipped, unmatched tiles
const unflippedTiles = [];
state.board.forEach(row => {
row.forEach(tile => {
if (tile && !tile.flipped && !tile.matched) {
unflippedTiles.push(tile);
}
});
});
if (unflippedTiles.length < 2) return;
// Find a matching pair
const tileCount = {};
let hintTile1 = null;
let hintTile2 = null;
for (const tile of unflippedTiles) {
if (tileCount[tile.type]) {
hintTile1 = tileCount[tile.type];
hintTile2 = tile;
break;
}
tileCount[tile.type] = tile;
}
if (hintTile1 && hintTile2) {
// Highlight the hint tiles
const tileElements = boardElement.querySelectorAll('.tile');
tileElements.forEach((element, index) => {
const row = Math.floor(index / 8);
const col = index % 8;
const tile = state.board[row]?.[col];
if (tile === hintTile1 || tile === hintTile2) {
element.classList.add('ring-4', 'ring-yellow-400', 'ring-opacity-75');
// Remove highlight after delay
setTimeout(() => {
element.classList.remove('ring-4', 'ring-yellow-400', 'ring-opacity-75');
}, 2000);
}
});
// Deduct points for using hint
state.score = Math.max(0, state.score - 25);
updateUI();
// Play hint sound
playSound('hint');
}
}
// Event listeners
newGameButton.addEventListener('click', initGame);
hintButton.addEventListener('click', provideHint);
soundToggleButton.addEventListener('click', () => {
state.soundEnabled = !state.soundEnabled;
soundToggleButton.innerHTML = state.soundEnabled
? '<i class="fas fa-volume-up mr-2"></i> Sound On'
: '<i class="fas fa-volume-mute mr-2"></i> Sound Off';
playSound('button-click');
});
nextLevelButton.addEventListener('click', nextLevel);
playAgainButton.addEventListener('click', () => {
playSound('button-click');
gameOverModal.classList.add('hidden');
initGame();
});
// Initialize the game
initGame();
});
</script>
</body>
</html>
-->
<!--<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>Vita Mahjong</title>
<script src="https://cdn.tailwindcss.com"></script>
<link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/font-awesome/6.4.0/css/all.min.css">
<style>
@keyframes tileHover {
0% { transform: translateY(0); }
50% { transform: translateY(-5px); }
100% { transform: translateY(0); }
}
.tile-hover:hover {
animation: tileHover 0.3s ease;
transform: translateY(-5px);
box-shadow: 0 10px 15px -3px rgba(0, 0, 0, 0.2);
}
.tile-selected {
transform: translateY(-15px);
box-shadow: 0 20px 25px -5px rgba(0, 0, 0, 0.2);
z-index: 10;
}
.tile-matched {
animation: fadeOut 0.5s ease forwards;
}
@keyframes fadeOut {
to {
opacity: 0;
transform: scale(0.8);
}
}
.board-container {
perspective: 1000px;
}
.tile {
transition: all 0.3s ease;
transform-style: preserve-3d;
}
.tile-inner {
position: relative;
width: 100%;
height: 100%;
transform-style: preserve-3d;
}
.tile-front, .tile-back {
position: absolute;
width: 100%;
height: 100%;
backface-visibility: hidden;
border-radius: 8px;
display: flex;
align-items: center;
justify-content: center;
}
.tile-front {
background: linear-gradient(145deg, #f0f0f0, #ffffff);
transform: rotateY(180deg);
}
.tile-back {
background: linear-gradient(145deg, #4f46e5, #7c3aed);
color: white;
}
.flipped {
transform: rotateY(180deg);
}
.level-complete {
animation: pulse 2s infinite;
}
@keyframes pulse {
0% { transform: scale(1); }
50% { transform: scale(1.05); }
100% { transform: scale(1); }
}
</style>
</head>
<body class="bg-gray-100 min-h-screen font-sans">
<div class="container mx-auto px-4 py-8">
<header class="flex justify-between items-center mb-8">
<div class="flex items-center">
<i class="fas fa-dragon text-4xl text-purple-600 mr-3"></i>
<h1 class="text-3xl font-bold text-gray-800">Vita Mahjong</h1>
</div>
<div class="flex items-center space-x-4">
<div class="bg-white rounded-lg shadow p-3 flex items-center">
<i class="fas fa-clock text-purple-600 mr-2"></i>
<span id="timer" class="font-bold">00:00</span>
</div>
<div class="bg-white rounded-lg shadow p-3 flex items-center">
<i class="fas fa-layer-group text-purple-600 mr-2"></i>
<span id="level" class="font-bold">Level 1</span>
</div>
<div class="bg-white rounded-lg shadow p-3 flex items-center">
<i class="fas fa-star text-yellow-500 mr-2"></i>
<span id="score" class="font-bold">0</span>
</div>
</div>
</header>
<div class="flex justify-between mb-6">
<div class="flex space-x-3">
<button id="new-game" class="bg-purple-600 hover:bg-purple-700 text-white px-4 py-2 rounded-lg shadow flex items-center">
<i class="fas fa-plus-circle mr-2"></i> New Game
</button>
<button id="hint" class="bg-amber-500 hover:bg-amber-600 text-white px-4 py-2 rounded-lg shadow flex items-center">
<i class="fas fa-lightbulb mr-2"></i> Hint
</button>
</div>
<div>
<button id="sound-toggle" class="bg-gray-200 hover:bg-gray-300 text-gray-800 px-4 py-2 rounded-lg shadow flex items-center">
<i class="fas fa-volume-up mr-2"></i> Sound On
</button>
</div>
</div>
<div class="board-container bg-white rounded-xl shadow-xl p-6 mb-6">
<div id="board" class="grid grid-cols-8 gap-3 mx-auto"></div>
</div>
<div id="game-status" class="text-center mb-6 hidden">
<div class="inline-block bg-green-100 border-l-4 border-green-500 text-green-700 p-4 rounded-lg">
<p class="font-bold">Level Complete!</p>
</div>
</div>
<div id="level-complete-modal" class="fixed inset-0 bg-black bg-opacity-50 flex items-center justify-center hidden z-50">
<div class="bg-white rounded-xl shadow-2xl p-8 max-w-md w-full level-complete">
<div class="text-center">
<div class="text-6xl text-yellow-500 mb-4">
<i class="fas fa-trophy"></i>
</div>
<h2 class="text-3xl font-bold text-gray-800 mb-2">Level Complete!</h2>
<p class="text-gray-600 mb-6">Great job! Ready for the next challenge?</p>
<div class="grid grid-cols-2 gap-4 mb-6">
<div class="bg-purple-50 rounded-lg p-3">
<p class="text-sm text-purple-600">Time</p>
<p id="level-time" class="font-bold text-xl">00:45</p>
</div>
<div class="bg-purple-50 rounded-lg p-3">
<p class="text-sm text-purple-600">Score</p>
<p id="level-score" class="font-bold text-xl">+250</p>
</div>
</div>
<button id="next-level" class="w-full bg-purple-600 hover:bg-purple-700 text-white py-3 rounded-lg shadow-lg font-bold">
Next Level <i class="fas fa-arrow-right ml-2"></i>
</button>
</div>
</div>
</div>
<div id="game-over-modal" class="fixed inset-0 bg-black bg-opacity-50 flex items-center justify-center hidden z-50">
<div class="bg-white rounded-xl shadow-2xl p-8 max-w-md w-full">
<div class="text-center">
<div class="text-6xl text-red-500 mb-4">
<i class="fas fa-gamepad"></i>
</div>
<h2 class="text-3xl font-bold text-gray-800 mb-2">Game Over</h2>
<p class="text-gray-600 mb-6">Better luck next time!</p>
<div class="grid grid-cols-2 gap-4 mb-6">
<div class="bg-purple-50 rounded-lg p-3">
<p class="text-sm text-purple-600">Level Reached</p>
<p id="final-level" class="font-bold text-xl">3</p>
</div>
<div class="bg-purple-50 rounded-lg p-3">
<p class="text-sm text-purple-600">Total Score</p>
<p id="final-score" class="font-bold text-xl">750</p>
</div>
</div>
<button id="play-again" class="w-full bg-purple-600 hover:bg-purple-700 text-white py-3 rounded-lg shadow-lg font-bold">
Play Again <i class="fas fa-redo ml-2"></i>
</button>
</div>
</div>
</div>
</div>
<script>
document.addEventListener('DOMContentLoaded', () => {
// Game state
const state = {
board: [],
level: 1,
score: 0,
time: 0,
timerInterval: null,
selectedTiles: [],
matchedPairs: 0,
totalPairs: 0,
soundEnabled: true,
gameActive: false
};
// DOM elements
const boardElement = document.getElementById('board');
const timerElement = document.getElementById('timer');
const levelElement = document.getElementById('level');
const scoreElement = document.getElementById('score');
const newGameButton = document.getElementById('new-game');
const hintButton = document.getElementById('hint');
const soundToggleButton = document.getElementById('sound-toggle');
const gameStatusElement = document.getElementById('game-status');
const levelCompleteModal = document.getElementById('level-complete-modal');
const gameOverModal = document.getElementById('game-over-modal');
const nextLevelButton = document.getElementById('next-level');
const playAgainButton = document.getElementById('play-again');
const levelTimeElement = document.getElementById('level-time');
const levelScoreElement = document.getElementById('level-score');
const finalLevelElement = document.getElementById('final-level');
const finalScoreElement = document.getElementById('final-score');
// Tile types (Vita Mahjong style)
const tileTypes = [
'1m', '2m', '3m', '4m', '5m', '6m', '7m', '8m', '9m', // Characters
'1s', '2s', '3s', '4s', '5s', '6s', '7s', '8s', '9s', // Bamboo
'1p', '2p', '3p', '4p', '5p', '6p', '7p', '8p', '9p', // Circles
'ew', 'sw', 'ww', 'nw', // Winds
'wd', 'gd', 'rd' // Dragons
];
// Initialize game
function initGame() {
state.level = 1;
state.score = 0;
state.time = 0;
state.matchedPairs = 0;
state.gameActive = true;
clearInterval(state.timerInterval);
startTimer();
updateUI();
createBoard();
}
// Create game board based on current level
function createBoard() {
boardElement.innerHTML = '';
state.board = [];
state.selectedTiles = [];
state.matchedPairs = 0;
// Determine number of pairs based on level
const pairs = Math.min(4 + state.level, 32); // Max 32 pairs (64 tiles)
state.totalPairs = pairs;
// Create array of tile pairs
let tiles = [];
const availableTypes = [...tileTypes].sort(() => 0.5 - Math.random()).slice(0, pairs);
availableTypes.forEach(type => {
tiles.push(type, type);
});
// Shuffle tiles
tiles = shuffleArray(tiles);
// Create board matrix (8 columns, rows as needed)
const cols = 8;
const rows = Math.ceil(tiles.length / cols);
for (let i = 0; i < rows; i++) {
const row = [];
for (let j = 0; j < cols; j++) {
const index = i * cols + j;
if (index < tiles.length) {
row.push({
type: tiles[index],
flipped: false,
matched: false,
row: i,
col: j
});
} else {
row.push(null); // Empty slot for uneven boards
}
}
state.board.push(row);
}
// Render tiles
renderBoard();
}
// Render the game board
function renderBoard() {
boardElement.innerHTML = '';
// Calculate grid rows based on board rows
boardElement.className = `grid gap-3 mx-auto`;
boardElement.style.gridTemplateColumns = `repeat(8, minmax(0, 1fr))`;
state.board.forEach((row, rowIndex) => {
row.forEach((tile, colIndex) => {
if (!tile) return;
const tileElement = document.createElement('div');
tileElement.className = `tile aspect-square cursor-pointer transition-all duration-300 ${tile.matched ? 'opacity-0' : ''}`;
tileElement.innerHTML = `
<div class="tile-inner ${tile.flipped ? 'flipped' : ''}">
<div class="tile-back flex items-center justify-center">
<i class="fas fa-dragon text-2xl"></i>
</div>
<div class="tile-front">
<span class="text-3xl font-bold">${getTileSymbol(tile.type)}</span>
</div>
</div>
`;
tileElement.addEventListener('click', () => handleTileClick(tile));
if (!tile.matched) {
tileElement.classList.add('tile-hover');
}
boardElement.appendChild(tileElement);
});
});
}
// Handle tile click
function handleTileClick(tile) {
if (!state.gameActive || tile.matched || tile.flipped || state.selectedTiles.length >= 2) {
return;
}
// Flip the tile
tile.flipped = true;
state.selectedTiles.push(tile);
// Play sound
if (state.soundEnabled) {
playSound('flip');
}
renderBoard();
// Check for match if two tiles are selected
if (state.selectedTiles.length === 2) {
const [tile1, tile2] = state.selectedTiles;
if (tile1.type === tile2.type) {
// Match found
tile1.matched = true;
tile2.matched = true;
state.matchedPairs++;
state.score += 50 * state.level;
// Play success sound
if (state.soundEnabled) {
playSound('match');
}
// Check if level is complete
if (state.matchedPairs === state.totalPairs) {
levelComplete();
}
// Clear selection after delay
setTimeout(() => {
state.selectedTiles = [];
renderBoard();
}, 500);
} else {
// No match
setTimeout(() => {
tile1.flipped = false;
tile2.flipped = false;
state.selectedTiles = [];
renderBoard();
// Play mismatch sound
if (state.soundEnabled) {
playSound('mismatch');
}
}, 1000);
}
}
updateUI();
}
// Level complete
function levelComplete() {
state.gameActive = false;
clearInterval(state.timerInterval);
// Calculate bonus points based on time
const timeBonus = Math.max(0, 300 - state.time);
state.score += timeBonus;
// Show level complete modal
levelTimeElement.textContent = formatTime(state.time);
levelScoreElement.textContent = `+${timeBonus}`;
levelCompleteModal.classList.remove('hidden');
updateUI();
}
// Next level
function nextLevel() {
state.level++;
state.time = 0;
state.gameActive = true;
levelCompleteModal.classList.add('hidden');
startTimer();
createBoard();
updateUI();
// Show level up message
gameStatusElement.classList.remove('hidden');
gameStatusElement.querySelector('p').textContent = `Level ${state.level}!`;
setTimeout(() => {
gameStatusElement.classList.add('hidden');
}, 2000);
}
// Game over
function gameOver() {
state.gameActive = false;
clearInterval(state.timerInterval);
// Update final stats
finalLevelElement.textContent = state.level;
finalScoreElement.textContent = state.score;
// Show game over modal
gameOverModal.classList.remove('hidden');
}
// Start timer
function startTimer() {
state.time = 0;
updateTimerDisplay();
state.timerInterval = setInterval(() => {
state.time++;
updateTimerDisplay();
// Game over if time exceeds limit (5 minutes)
if (state.time >= 300) {
gameOver();
}
}, 1000);
}
// Update timer display
function updateTimerDisplay() {
timerElement.textContent = formatTime(state.time);
}
// Format time as MM:SS
function formatTime(seconds) {
const mins = Math.floor(seconds / 60).toString().padStart(2, '0');
const secs = (seconds % 60).toString().padStart(2, '0');
return `${mins}:${secs}`;
}
// Update UI elements
function updateUI() {
levelElement.textContent = `Level ${state.level}`;
scoreElement.textContent = state.score;
}
// Get tile symbol for display
function getTileSymbol(type) {
const symbols = {
'1m': '一', '2m': '二', '3m': '三', '4m': '四', '5m': '五',
'6m': '六', '7m': '七', '8m': '八', '9m': '九',
'1s': '1', '2s': '2', '3s': '3', '4s': '4', '5s': '5',
'6s': '6', '7s': '7', '8s': '8', '9s': '9',
'1p': '❶', '2p': '❷', '3p': '❸', '4p': '❹', '5p': '❺',
'6p': '❻', '7p': '❼', '8p': '❽', '9p': '❾',
'ew': '東', 'sw': '南', 'ww': '西', 'nw': '北',
'wd': '白', 'gd': '發', 'rd': '中'
};
return symbols[type] || type;
}
// Shuffle array
function shuffleArray(array) {
const newArray = [...array];
for (let i = newArray.length - 1; i > 0; i--) {
const j = Math.floor(Math.random() * (i + 1));
[newArray[i], newArray[j]] = [newArray[j], newArray[i]];
}
return newArray;
}
// Play sound
function playSound(type) {
// In a real implementation, you would play actual sound files
console.log(`Playing ${type} sound`);
}
// Provide hint
function provideHint() {
if (!state.gameActive || state.matchedPairs === state.totalPairs) {
return;
}
// Find all unflipped, unmatched tiles
const unflippedTiles = [];
state.board.forEach(row => {
row.forEach(tile => {
if (tile && !tile.flipped && !tile.matched) {
unflippedTiles.push(tile);
}
});
});
if (unflippedTiles.length < 2) return;
// Find a matching pair
const tileCount = {};
let hintTile1 = null;
let hintTile2 = null;
for (const tile of unflippedTiles) {
if (tileCount[tile.type]) {
hintTile1 = tileCount[tile.type];
hintTile2 = tile;
break;
}
tileCount[tile.type] = tile;
}
if (hintTile1 && hintTile2) {
// Highlight the hint tiles
const tileElements = boardElement.querySelectorAll('.tile');
tileElements.forEach((element, index) => {
const row = Math.floor(index / 8);
const col = index % 8;
const tile = state.board[row]?.[col];
if (tile === hintTile1 || tile === hintTile2) {
element.classList.add('ring-4', 'ring-yellow-400', 'ring-opacity-75');
// Remove highlight after delay
setTimeout(() => {
element.classList.remove('ring-4', 'ring-yellow-400', 'ring-opacity-75');
}, 2000);
}
});
// Deduct points for using hint
state.score = Math.max(0, state.score - 25);
updateUI();
// Play hint sound
if (state.soundEnabled) {
playSound('hint');
}
}
}
// Event listeners
newGameButton.addEventListener('click', initGame);
hintButton.addEventListener('click', provideHint);
soundToggleButton.addEventListener('click', () => {
state.soundEnabled = !state.soundEnabled;
soundToggleButton.innerHTML = state.soundEnabled
? '<i class="fas fa-volume-up mr-2"></i> Sound On'
: '<i class="fas fa-volume-mute mr-2"></i> Sound Off';
});
nextLevelButton.addEventListener('click', nextLevel);
playAgainButton.addEventListener('click', () => {
gameOverModal.classList.add('hidden');
initGame();
});
// Initialize the game
initGame();
});
</script>
</body>
</html>
-->