MediaWiki:Common.js: Unterschied zwischen den Versionen
Admin (Diskussion | Beiträge) Keine Bearbeitungszusammenfassung Markierung: Manuelle Zurücksetzung |
Admin (Diskussion | Beiträge) Keine Bearbeitungszusammenfassung |
||
| Zeile 739: | Zeile 739: | ||
// ------------------------------------- | // ------------------------------------- | ||
/* Whisky News | /* Whisky News Popup mit Slàinte mhath Animation (v6) */ | ||
*/ | |||
mw.loader.using(['mediawiki.util','jquery']).then(function(){ | mw.loader.using(['mediawiki.util','jquery']).then(function(){ | ||
(function($, mw){ | (function($, mw){ | ||
| Zeile 750: | Zeile 746: | ||
var CONFIG = { | var CONFIG = { | ||
enabled: true, | enabled: true, | ||
id: 'wow_mannheim_whisky_news_v6', | id: 'wow_mannheim_whisky_news_v6', | ||
title: 'Whisky News: Messeabfüllungen – World of Whisky (Mannheim)', | title: 'Whisky News: Messeabfüllungen – World of Whisky (Mannheim)', | ||
introHTML: '<p>Frisch zur Messe in Mannheim: Zwei limitierte Abfüllungen. Schau sie dir an und bewerte sie im Wiki!</p>', | introHTML: '<p>Frisch zur Messe in Mannheim: Zwei limitierte Abfüllungen. Schau sie dir an und bewerte sie im Wiki!</p>', | ||
images: [ | images: [ | ||
{ | { | ||
src: 'https://ados-wiki.de/images/2/2f/South_Islay_Single_Malt_13_year-old_%28Sherry_Octave_Cask_Finish%29.single.jpg', | src: 'https://ados-wiki.de/images/2/2f/South_Islay_Single_Malt_13_year-old_%28Sherry_Octave_Cask_Finish%29.single.jpg', | ||
alt: 'South Islay 13y – Sherry Octave Cask Finish', | alt: 'South Islay 13y – Sherry Octave Cask Finish', | ||
link: 'https://ados-wiki.de/wiki/South_Islay_Single_Malt_13_year-old_(Sherry_Octave_Cask_Finish)', | link: 'https://ados-wiki.de/wiki/South_Islay_Single_Malt_13_year-old_(Sherry_Octave_Cask_Finish)', | ||
| Zeile 763: | Zeile 757: | ||
}, | }, | ||
{ | { | ||
src: 'https://ados-wiki.de/images/9/95/Tullibardine_13_year-old_%28Shiraz_Wine_Octave_Cask_Finish%29.single.jpg', | src: 'https://ados-wiki.de/images/9/95/Tullibardine_13_year-old_%28Shiraz_Wine_Octave_Cask_Finish%29.single.jpg', | ||
alt: 'Tullibardine 13y – Shiraz Wine Octave Cask Finish', | alt: 'Tullibardine 13y – Shiraz Wine Octave Cask Finish', | ||
link: 'https://ados-wiki.de/wiki/Tullibardine_13_year-old', | link: 'https://ados-wiki.de/wiki/Tullibardine_13_year-old', | ||
| Zeile 769: | Zeile 763: | ||
} | } | ||
], | ], | ||
toastText: 'Slàinte mhath', | |||
dailyLimit: 1, | dailyLimit: 1, | ||
escToClose: true, | escToClose: true, | ||
clickBackdropToClose: true | clickBackdropToClose: true | ||
}; | }; | ||
if (!CONFIG.enabled) return; | if (!CONFIG.enabled) return; | ||
// | // === LocalStorage: nur 1x pro Tag === | ||
var isAnon = (mw.config.get('wgUserName') === null); | var isAnon = (mw.config.get('wgUserName') === null); | ||
function LSget(k){ try { return localStorage.getItem(k); } catch(e){ return null; } } | function LSget(k){ try { return localStorage.getItem(k); } catch(e){ return null; } } | ||
function LSset(k,v){ try { localStorage.setItem(k, v); } catch(e){} } | function LSset(k,v){ try { localStorage.setItem(k,v); } catch(e){} } | ||
var key = 'popup_' + CONFIG.id + (isAnon?':anon':':user'); | var key = 'popup_' + CONFIG.id + (isAnon?':anon':':user'); | ||
var today = (d => d.getFullYear()+'-'+('0'+(d.getMonth()+1)).slice(-2)+'-'+('0'+d.getDate()).slice(-2))(new Date()); | var today = (d => d.getFullYear()+'-'+('0'+(d.getMonth()+1)).slice(-2)+'-'+('0'+d.getDate()).slice(-2))(new Date()); | ||
| Zeile 791: | Zeile 781: | ||
$(function(){ | $(function(){ | ||
// | // === Popup Grundstruktur === | ||
var $overlay = $('<div>', {'class':'mw-popup-overlay'}); | var $overlay = $('<div>', {'class':'mw-popup-overlay'}); | ||
var $modal = $('<div>', { | var $modal = $('<div>', {'class':'mw-popup-modal','role':'dialog','aria-modal':'true'}); | ||
var $stage = $('<div>', {'class':'mw-fw-canvas-wrap'}); | var $stage = $('<div>', {'class':'mw-fw-canvas-wrap'}); | ||
var $canvas = $('<canvas>', {'class':'mw-fw-canvas','aria-hidden':'true'}); | var $canvas = $('<canvas>', {'class':'mw-fw-canvas','aria-hidden':'true'}); | ||
$stage.append($canvas); | $stage.append($canvas); | ||
var $toast = $('<div>', {'class':'mw-slainte-toast'}).text(CONFIG.toastText); | |||
var $toast = $('<div>', {'class':'mw-slainte-toast | |||
var $title = $('<h2>').text(CONFIG.title); | var $title = $('<h2>').text(CONFIG.title); | ||
var $intro = $('<div>', {'class':'mw-popup-content'}).html(CONFIG.introHTML); | var $intro = $('<div>', {'class':'mw-popup-content'}).html(CONFIG.introHTML); | ||
// Karten | // === Karten === | ||
var $cards = $('<div>', {'class':'mw-wnews-cards'}); | var $cards = $('<div>', {'class':'mw-wnews-cards'}); | ||
CONFIG.images.forEach(function(img){ | CONFIG.images.forEach(function(img){ | ||
| Zeile 828: | Zeile 809: | ||
}); | }); | ||
// Button | // === Button === | ||
var $btnRow = $('<div>', {'class':'mw-popup-button-row'}); | var $btnRow = $('<div>', {'class':'mw-popup-button-row'}); | ||
var $ok = $('<button>', {'class':'mw-popup-close', type:'button'}).text('OK'); | var $ok = $('<button>', {'class':'mw-popup-close', type:'button'}).text('OK'); | ||
$btnRow.append($ok); | $btnRow.append($ok); | ||
// | // === Zusammenbauen === | ||
$modal.append($stage, $toast, $title, $intro, $cards, $btnRow); | $modal.append($stage, $toast, $title, $intro, $cards, $btnRow); | ||
$('body').append($overlay, $modal); | $('body').append($overlay, $modal); | ||
// === Schließen === | |||
function close(){ | function close(){ | ||
stopAnim(true); | stopAnim(true); | ||
| Zeile 852: | Zeile 834: | ||
} | } | ||
// | // === Canvas Whiskyglas Animation === | ||
var canvas = $canvas[0], ctx = canvas.getContext && canvas.getContext('2d'); | var canvas = $canvas[0], ctx = canvas.getContext && canvas.getContext('2d'); | ||
var reduce = window.matchMedia && window.matchMedia('(prefers-reduced-motion: reduce)').matches; | var reduce = window.matchMedia && window.matchMedia('(prefers-reduced-motion: reduce)').matches; | ||
var dpr=1, cw=0, ch=0, raf=null, t0=0, started=false, ro; | var dpr=1, cw=0, ch=0, raf=null, t0=0, started=false, ro; | ||
var glass = {x:0,y:0,w:0,h:0,r:0} | var glass={x:0,y:0,w:0,h:0,r:0}, baseFill=0, amp=0; | ||
function setSize(){ | function setSize(){ | ||
| Zeile 875: | Zeile 856: | ||
function layout(){ | function layout(){ | ||
var w=canvas.width, h=canvas.height; | var w=canvas.width, h=canvas.height; | ||
glass.x = w*0.18; glass.y = h*0.15; glass.w = w*0.64; glass.h = h*0.7; | glass.x=w*0.18; glass.y=h*0.15; glass.w=w*0.64; glass.h=h*0.7; | ||
glass.r = Math.min(glass.w,glass.h)*0.08; | glass.r=Math.min(glass.w,glass.h)*0.08; | ||
baseFill = glass.y + glass.h*0.58; | baseFill=glass.y+glass.h*0.58; amp=Math.min(14*dpr,canvas.height*0.03); | ||
} | } | ||
function drawGlass(){ | function drawGlass(){ | ||
ctx.save(); | ctx.save(); | ||
ctx.strokeStyle = 'rgba(255,255,255,0.28)'; | ctx.strokeStyle='rgba(255,255,255,0.28)'; | ||
ctx.lineWidth = Math.max(2, 2*dpr); | ctx.lineWidth=Math.max(2,2*dpr); | ||
ctx.beginPath(); | ctx.beginPath(); | ||
ctx.moveTo(glass.x+glass.r, glass.y); | ctx.moveTo(glass.x+glass.r,glass.y); | ||
ctx.arcTo(glass.x+glass.w, glass.y, glass.x+glass.w, glass.y+glass.h, glass.r); | ctx.arcTo(glass.x+glass.w,glass.y,glass.x+glass.w,glass.y+glass.h,glass.r); | ||
ctx.arcTo(glass.x+glass.w, glass.y+glass.h, glass.x, glass.y+glass.h, glass.r); | ctx.arcTo(glass.x+glass.w,glass.y+glass.h,glass.x,glass.y+glass.h,glass.r); | ||
ctx.arcTo(glass.x, glass.y+glass.h, glass.x, glass.y, glass.r); | ctx.arcTo(glass.x,glass.y+glass.h,glass.x,glass.y,glass.r); | ||
ctx.arcTo(glass.x, glass.y, glass.x+glass.w, glass.y, glass.r); | ctx.arcTo(glass.x,glass.y,glass.x+glass.w,glass.y,glass.r); | ||
ctx.closePath(); | ctx.closePath(); ctx.stroke(); | ||
ctx.beginPath(); | ctx.beginPath(); | ||
ctx.moveTo(glass.x+glass.w*0.15, glass.y+glass.h*0.1); | ctx.moveTo(glass.x+glass.w*0.15,glass.y+glass.h*0.1); | ||
ctx.quadraticCurveTo(glass.x+glass.w*0.25, glass.y+glass.h*0.05, glass.x+glass.w*0.3, glass.y+glass.h*0.3); | ctx.quadraticCurveTo(glass.x+glass.w*0.25,glass.y+glass.h*0.05,glass.x+glass.w*0.3,glass.y+glass.h*0.3); | ||
ctx.strokeStyle = 'rgba(255,255,255,0.18)'; | ctx.strokeStyle='rgba(255,255,255,0.18)'; ctx.stroke(); ctx.restore(); | ||
} | } | ||
function drawLiquid(t){ | function drawLiquid(t){ | ||
var topY = baseFill + Math.sin(t*2.0)*amp*0.25 + Math.sin(t*0.7)*amp*0.15; | var topY=baseFill+Math.sin(t*2.0)*amp*0.25+Math.sin(t*0.7)*amp*0.15; | ||
ctx.save(); | ctx.save(); | ||
ctx.beginPath(); | ctx.beginPath(); | ||
ctx.moveTo(glass.x+glass.r, glass.y); | ctx.moveTo(glass.x+glass.r,glass.y); | ||
ctx.arcTo(glass.x+glass.w, glass.y, glass.x+glass.w, glass.y+glass.h, glass.r); | ctx.arcTo(glass.x+glass.w,glass.y,glass.x+glass.w,glass.y+glass.h,glass.r); | ||
ctx.arcTo(glass.x+glass.w, glass.y+glass.h, glass.x, glass.y+glass.h, glass.r); | ctx.arcTo(glass.x+glass.w,glass.y+glass.h,glass.x,glass.y+glass.h,glass.r); | ||
ctx.arcTo(glass.x, glass.y+glass.h, glass.x, glass.y, glass.r); | ctx.arcTo(glass.x,glass.y+glass.h,glass.x,glass.y,glass.r); | ||
ctx.arcTo(glass.x, glass.y, glass.x+glass.w, glass.y, glass.r); | ctx.arcTo(glass.x,glass.y,glass.x+glass.w,glass.y,glass.r); | ||
ctx.closePath(); | ctx.closePath(); ctx.clip(); | ||
var grd=ctx.createLinearGradient(0,topY-30*dpr,0,glass.y+glass.h); | |||
grd.addColorStop(0,'rgba(255,190,90,0.96)'); | |||
grd.addColorStop(1,'rgba(170,85,20,0.98)'); | |||
var grd = ctx.createLinearGradient(0, topY-30*dpr, 0, glass.y+glass.h); | |||
grd.addColorStop(0, 'rgba(255,190,90,0.96)'); | |||
grd.addColorStop(1, 'rgba(170,85,20,0.98)'); | |||
ctx.beginPath(); | ctx.beginPath(); | ||
ctx.moveTo(glass.x, topY); | ctx.moveTo(glass.x,topY); | ||
for (var x=0; x<=glass.w; x+=6*dpr){ | for(var x=0;x<=glass.w;x+=6*dpr){ | ||
var y = topY + Math.sin((x*0.055) + t*3.6) * amp * 0.22; | var y=topY+Math.sin((x*0.055)+t*3.6)*amp*0.22; | ||
ctx.lineTo(glass.x + x, y); | ctx.lineTo(glass.x+x,y); | ||
} | } | ||
ctx.lineTo(glass.x + glass.w, glass.y + glass.h); | ctx.lineTo(glass.x+glass.w,glass.y+glass.h); | ||
ctx.lineTo(glass.x, | ctx.lineTo(glass.x,glass.y+glass.h); | ||
ctx.closePath(); | ctx.closePath(); | ||
ctx.fillStyle = grd; | ctx.fillStyle=grd; ctx.fill(); | ||
ctx.beginPath(); | ctx.beginPath(); | ||
ctx.moveTo(glass.x, topY); | ctx.moveTo(glass.x,topY); | ||
for (var x2=0; x2<=glass.w; x2+=6*dpr){ | for(var x2=0;x2<=glass.w;x2+=6*dpr){ | ||
var y2 = topY + Math.sin((x2*0.055) + t*3.6) * amp * 0.22; | var y2=topY+Math.sin((x2*0.055)+t*3.6)*amp*0.22; | ||
ctx.lineTo(glass.x + x2, y2); | ctx.lineTo(glass.x+x2,y2); | ||
} | } | ||
ctx.strokeStyle = 'rgba(255,215,120,0.35)'; | ctx.strokeStyle='rgba(255,215,120,0.35)'; | ||
ctx.lineWidth = Math.max(1, 1*dpr); | ctx.lineWidth=Math.max(1,1*dpr); ctx.stroke(); | ||
ctx.restore(); | ctx.restore(); | ||
} | } | ||
function frame(ts){ | function frame(ts){ | ||
if (!t0) t0 = ts; | if(!t0) t0=ts; | ||
var t = (ts - t0)/1000; | var t=(ts-t0)/1000; | ||
ctx.globalCompositeOperation='source-over'; | |||
ctx.fillStyle='rgba(5,10,20,0.16)'; | |||
ctx.globalCompositeOperation = 'source-over'; | |||
ctx.fillStyle = 'rgba(5,10,20,0.16)'; | |||
ctx.fillRect(0,0,canvas.width,canvas.height); | ctx.fillRect(0,0,canvas.width,canvas.height); | ||
drawGlass(); drawLiquid(t); | |||
drawGlass(); | |||
ctx.save(); | ctx.save(); | ||
ctx.globalCompositeOperation = 'lighter'; | ctx.globalCompositeOperation='lighter'; | ||
ctx.strokeStyle = 'rgba(255,255,255,0.15)'; | ctx.strokeStyle='rgba(255,255,255,0.15)'; | ||
ctx.lineWidth = 1*dpr; | ctx.lineWidth=1*dpr; | ||
ctx.beginPath(); | ctx.beginPath(); | ||
ctx.arc(glass.x + glass.w*0.85, glass.y + glass.h*0.15, 10*dpr, 0, Math.PI*2); | ctx.arc(glass.x+glass.w*0.85,glass.y+glass.h*0.15,10*dpr,0,Math.PI*2); | ||
ctx.stroke(); | ctx.stroke(); ctx.restore(); | ||
raf=requestAnimationFrame(frame); | |||
raf = requestAnimationFrame(frame); | |||
} | } | ||
function startAnim(){ | function startAnim(){ | ||
if (!ctx || reduce) return; | if(!ctx||reduce)return; | ||
if (!setSize()) { setTimeout(startAnim, 50); return; } | if(!setSize()){setTimeout(startAnim,50);return;} | ||
if (started) return; | if(started)return; | ||
started = true; t0=0; | started=true;t0=0; | ||
raf = requestAnimationFrame(frame); | raf=requestAnimationFrame(frame); | ||
if('ResizeObserver'in window){ | |||
if ('ResizeObserver' in window) { | ro=new ResizeObserver(function(){setSize();}); | ||
ro = new ResizeObserver(function(){ setSize(); }); | |||
ro.observe($stage[0]); | ro.observe($stage[0]); | ||
} else { | }else{$(window).on('resize.mwwnews',setSize);} | ||
document.addEventListener('visibilitychange',onVis); | |||
document.addEventListener('visibilitychange', onVis); | |||
} | } | ||
function stopAnim(remove){ | function stopAnim(remove){ | ||
if (raf){ cancelAnimationFrame(raf); raf=null; } | if(raf){cancelAnimationFrame(raf);raf=null;} | ||
started = false; | started=false; | ||
if (remove){ | if(remove){ | ||
if (ro){ ro.disconnect(); ro=null; } else { $(window).off('resize.mwwnews'); } | if(ro){ro.disconnect();ro=null;}else{$(window).off('resize.mwwnews');} | ||
document.removeEventListener('visibilitychange', onVis); | document.removeEventListener('visibilitychange',onVis); | ||
} | } | ||
} | } | ||
function onVis(){ if (document.hidden) stopAnim(false); else startAnim(); } | function onVis(){if(document.hidden)stopAnim(false);else startAnim();} | ||
setTimeout(startAnim,0); | |||
setTimeout(startAnim, 0); | |||
markSeen(); | markSeen(); | ||
}); | }); | ||
})(jQuery, mw); | })(jQuery,mw); | ||
}); | }); | ||