« Module:Episode/MultiSeries » : différence entre les versions
Apparence
Contenu supprimé Contenu ajouté
Aucun résumé des modifications |
m LIMAFOX76 a déplacé la page Module:Episode REAL/MultiSeries vers Module:Episode/MultiSeries sans laisser de redirection |
||
| (19 versions intermédiaires par le même utilisateur non affichées) | |||
| Ligne 1 : | Ligne 1 : | ||
-- Module:Episode/MultiSeries |
|||
-- Fusionne toutes les séries et saisons (SG-1, SGA, SGU, Films…) |
|||
-- Fournit : get(id), get_by_alias(alias), resolve_alias(alias) |
|||
-- + Détection automatique des doublons |
|||
-- + Cache interne pour éviter le signal 24 |
|||
------------------------------------------------------------ |
------------------------------------------------------------ |
||
-- Module:Episode/MultiSeries (VERSION FINALE + CACHE INTERNE) |
|||
-- CACHE : si déjà chargé, on renvoie la version en mémoire |
|||
-- Fusion SG1 + SGA + SGU + SGI + SGO + FILMS |
|||
-- Normalisation, alias, indexation, cache interne |
|||
------------------------------------------------------------ |
------------------------------------------------------------ |
||
local cached = package.loaded["Module:Episode/MultiSeries"] |
|||
-- 0. CACHE INTERNE : si déjà chargé, renvoyer immédiatement |
|||
if type(cached) == "table" then |
|||
local loaded = package.loaded["Module:Episode/MultiSeries"] |
|||
return cached |
|||
if loaded then |
|||
return loaded |
|||
end |
end |
||
local Multi = {} |
|||
------------------------------------------------------------ |
|||
-- Dépendances |
|||
------------------------------------------------------------ |
|||
local utils = require("Module:Episode/Shared/Utils") |
|||
------------------------------------------------------------ |
------------------------------------------------------------ |
||
-- 1. Chargement des séries |
|||
-- Structure principale |
|||
------------------------------------------------------------ |
------------------------------------------------------------ |
||
local SG1 = require("Module:Episode/SG1/Index") |
|||
local Multi = { |
|||
local SGA = require("Module:Episode/SGA/Index") |
|||
episodes = {}, -- id → table épisode |
|||
local SGU = require("Module:Episode/SGU/Index") |
|||
aliases = {}, -- alias normalisé → id |
|||
local SGI = require("Module:Episode/SGI/Index") |
|||
loaded = {}, -- modules chargés |
|||
local SGO = require("Module:Episode/SGO/Index") |
|||
duplicates = nil -- rapport de doublons |
|||
local FILMS = require("Module:Episode/Films/Index") |
|||
} |
|||
------------------------------------------------------------ |
------------------------------------------------------------ |
||
-- 2. Tables globales |
|||
-- Liste des modules index de séries |
|||
-- (tu ajoutes ici SG1, SGA, SGU, Films, etc.) |
|||
------------------------------------------------------------ |
------------------------------------------------------------ |
||
Multi.episodes = {} |
|||
local SERIES_INDEX = { |
|||
Multi.by_id = {} |
|||
"Module:Episode/SG1/Index", |
|||
Multi.by_alias = {} |
|||
"Module:Episode/SGA/Index", |
|||
Multi.by_page = {} |
|||
"Module:Episode/SGU/Index", |
|||
Multi.by_series = {} |
|||
"Module:Episode/Films/Index", |
|||
Multi.by_season = {} |
|||
} |
|||
------------------------------------------------------------ |
------------------------------------------------------------ |
||
-- 3. Fonction normalize() exposée |
|||
-- Charge un module de saison et fusionne son contenu |
|||
------------------------------------------------------------ |
------------------------------------------------------------ |
||
local function |
local function normalize(str) |
||
if not str then return "" end |
|||
if not ok or type(mod) ~= "table" then |
|||
return false |
|||
end |
|||
str = str |
|||
for key, value in pairs(mod) do |
|||
:gsub("œ", "oe"):gsub("Œ", "oe") |
|||
:gsub("æ", "ae"):gsub("Æ", "ae") |
|||
:gsub("ç", "c"):gsub("Ç", "c") |
|||
:gsub("é", "e"):gsub("É", "e") |
|||
:gsub("è", "e"):gsub("È", "e") |
|||
:gsub("ê", "e"):gsub("Ê", "e") |
|||
:gsub("ë", "e"):gsub("Ë", "e") |
|||
:gsub("à", "a"):gsub("À", "a") |
|||
:gsub("â", "a"):gsub("Â", "a") |
|||
:gsub("ä", "a"):gsub("Ä", "a") |
|||
:gsub("î", "i"):gsub("Î", "i") |
|||
:gsub("ï", "i"):gsub("Ï", "i") |
|||
:gsub("ô", "o"):gsub("Ô", "o") |
|||
:gsub("ö", "o"):gsub("Ö", "o") |
|||
:gsub("ù", "u"):gsub("Ù", "u") |
|||
:gsub("û", "u"):gsub("Û", "u") |
|||
:gsub("ü", "u"):gsub("Ü", "u") |
|||
str = mw.ustring.lower(str) |
|||
------------------------------------------------------------ |
|||
str = mw.ustring.toNFD(str) |
|||
-- Cas 1 : entrée d’épisode (table avec .id) |
|||
str = mw.ustring.gsub(str, "%pM", "") |
|||
------------------------------------------------------------ |
|||
str = str:gsub("'%s*", "") |
|||
if type(value) == "table" and value.id then |
|||
str = str:gsub("[{}]", " ") |
|||
Multi.episodes[value.id] = value |
|||
str = str:gsub("[!\"#%$%%&%(%)*+,%.%/%:;%<%=%>%?%@%[%]%^_%`%{%|%}%~%-]", " ") |
|||
str = str:gsub("[^%w%s]", " ") |
|||
-- Ajout des alias déclarés dans l’épisode |
|||
str = str:gsub("%s+", " ") |
|||
if type(value.aliases) == "table" then |
|||
str = mw.text.trim(str) |
|||
for _, alias in ipairs(value.aliases) do |
|||
local norm = utils.normalize_alias(alias) |
|||
if norm then |
|||
Multi.aliases[norm] = value.id |
|||
end |
|||
end |
|||
end |
|||
------------------------------------------------------------ |
|||
-- Cas 2 : alias → id (clé = alias, valeur = id) |
|||
------------------------------------------------------------ |
|||
elseif type(value) == "string" then |
|||
local norm = utils.normalize_alias(key) |
|||
if norm then |
|||
Multi.aliases[norm] = value |
|||
end |
|||
end |
|||
end |
|||
return str |
|||
table.insert(Multi.loaded, path) |
|||
return true |
|||
end |
end |
||
Multi.normalize = normalize |
|||
------------------------------------------------------------ |
------------------------------------------------------------ |
||
-- 4. Alias techniques SGX |
|||
-- Charge un module index de série (liste des saisons) |
|||
------------------------------------------------------------ |
------------------------------------------------------------ |
||
local function |
local function make_tech_aliases(series, season, episode) |
||
if not season or not episode then |
|||
local ok, idx = pcall(require, path) |
|||
return {} -- films → aucun alias technique |
|||
if not ok or type(idx) ~= "table" or not idx.list_seasons then |
|||
return |
|||
end |
end |
||
local season_num = tonumber(season) |
|||
for _, entry in ipairs(idx.list_seasons()) do |
|||
local e = string.format("%02d", tonumber(episode)) |
|||
load_season_module(entry.module) |
|||
end |
|||
local s1 = string.format("%02d", season_num) |
|||
local s2 = tostring(season_num) |
|||
local s3 = tostring(season_num) |
|||
series = normalize(series) |
|||
return { |
|||
string.format("%s s%se%s", series, s1, e), |
|||
string.format("%s %sx%s", series, s2, e), |
|||
string.format("%s %s%s", series, s3, e), |
|||
} |
|||
end |
end |
||
------------------------------------------------------------ |
------------------------------------------------------------ |
||
-- 5. Génération complète des alias |
|||
-- Détection des doublons (IDs, alias, alias contradictoires) |
|||
------------------------------------------------------------ |
------------------------------------------------------------ |
||
local function |
local function make_aliases(ep) |
||
local |
local a = {} |
||
duplicate_ids = {}, |
|||
table.insert(a, normalize(ep.page_title_fr)) |
|||
duplicate_aliases = {}, |
|||
table.insert(a, normalize(ep.page_title_en)) |
|||
conflicting_aliases = {}, |
|||
table.insert(a, normalize(ep.id)) |
|||
} |
|||
if ep.season and ep.episode then |
|||
-- IDs |
|||
for _, alias in ipairs(make_tech_aliases(ep.series, ep.season, ep.episode)) do |
|||
local seen_ids = {} |
|||
table.insert(a, alias) |
|||
if seen_ids[id] then |
|||
table.insert(report.duplicate_ids, id) |
|||
else |
|||
seen_ids[id] = true |
|||
end |
end |
||
end |
end |
||
if ep.extra_aliases then |
|||
-- Aliases |
|||
for _, alias in ipairs(ep.extra_aliases) do |
|||
local alias_targets = {} |
|||
table.insert(a, normalize(alias)) |
|||
for alias, id in pairs(Multi.aliases) do |
|||
if alias_targets[alias] and alias_targets[alias] ~= id then |
|||
table.insert(report.conflicting_aliases, { |
|||
alias = alias, |
|||
id1 = alias_targets[alias], |
|||
id2 = id, |
|||
}) |
|||
elseif alias_targets[alias] then |
|||
table.insert(report.duplicate_aliases, alias) |
|||
else |
|||
alias_targets[alias] = id |
|||
end |
end |
||
end |
end |
||
return a |
|||
Multi.duplicates = report |
|||
end |
end |
||
------------------------------------------------------------ |
------------------------------------------------------------ |
||
-- 6. Ajout d’une série complète |
|||
-- Chargement global (toutes séries + saisons) |
|||
------------------------------------------------------------ |
------------------------------------------------------------ |
||
local function |
local function append_series(series_module) |
||
for _, |
for _, ep in ipairs(series_module.episodes or {}) do |
||
load_series_index(series_path) |
|||
end |
|||
detect_duplicates() |
|||
end |
|||
ep.aliases = make_aliases(ep) |
|||
load_all() |
|||
Multi.by_id[ep.id] = ep |
|||
------------------------------------------------------------ |
|||
-- API publique |
|||
------------------------------------------------------------ |
|||
if ep.page_title_fr then |
|||
-- Récupère un épisode par ID |
|||
Multi.by_page[normalize(ep.page_title_fr)] = ep |
|||
function Multi.get(id) |
|||
end |
|||
return Multi.episodes[id] |
|||
if ep.page_title_en then |
|||
end |
|||
Multi.by_page[normalize(ep.page_title_en)] = ep |
|||
end |
|||
for _, alias in ipairs(ep.aliases) do |
|||
-- Résout un alias → ID |
|||
alias = normalize(alias) |
|||
function Multi.resolve_alias(alias) |
|||
Multi.by_alias[alias] = Multi.by_alias[alias] or {} |
|||
table.insert(Multi.by_alias[alias], ep) |
|||
end |
|||
return Multi.aliases[norm] |
|||
end |
|||
Multi.by_series[ep.series] = Multi.by_series[ep.series] or {} |
|||
-- Récupère un épisode via alias |
|||
table.insert(Multi.by_series[ep.series], ep) |
|||
function Multi.get_by_alias(alias) |
|||
local id = Multi.resolve_alias(alias) |
|||
if |
if ep.season then |
||
local key = ep.series .. "-" .. ep.season |
|||
return Multi.get(id) |
|||
Multi.by_season[key] = Multi.by_season[key] or {} |
|||
table.insert(Multi.by_season[key], ep) |
|||
end |
|||
table.insert(Multi.episodes, ep) |
|||
end |
end |
||
return nil |
|||
end |
end |
||
------------------------------------------------------------ |
------------------------------------------------------------ |
||
-- 7. Fusion des séries (UNE SEULE FOIS) |
|||
-- CACHE : on stocke la table Multi pour les appels suivants |
|||
------------------------------------------------------------ |
|||
append_series(SG1) |
|||
append_series(SGA) |
|||
append_series(SGU) |
|||
append_series(SGI) |
|||
append_series(SGO) |
|||
append_series(FILMS) |
|||
------------------------------------------------------------ |
|||
-- 8. Enregistrer dans package.loaded (cache interne) |
|||
------------------------------------------------------------ |
------------------------------------------------------------ |
||
package.loaded["Module:Episode/MultiSeries"] = Multi |
package.loaded["Module:Episode/MultiSeries"] = Multi |
||
function Multi.debug() |
|||
return "MultiSeries chargé OK" |
|||
end |
|||
return Multi |
return Multi |
||
Dernière version du 3 juin 2026 à 14:15
| Il sera peut-être nécessaire de créer une page documentation pour ce module Scribunto programmé dans la page créer Les éditeurs peuvent faire des tests sur les pages bac à sable (créer | miroir) et études de cas (créer) du module. Veuillez ajouter les catégories dans la sous-page /documentation. Sous-pages de ce module. |
------------------------------------------------------------
-- Module:Episode/MultiSeries (VERSION FINALE + CACHE INTERNE)
-- Fusion SG1 + SGA + SGU + SGI + SGO + FILMS
-- Normalisation, alias, indexation, cache interne
------------------------------------------------------------
-- 0. CACHE INTERNE : si déjà chargé, renvoyer immédiatement
local loaded = package.loaded["Module:Episode/MultiSeries"]
if loaded then
return loaded
end
local Multi = {}
------------------------------------------------------------
-- 1. Chargement des séries
------------------------------------------------------------
local SG1 = require("Module:Episode/SG1/Index")
local SGA = require("Module:Episode/SGA/Index")
local SGU = require("Module:Episode/SGU/Index")
local SGI = require("Module:Episode/SGI/Index")
local SGO = require("Module:Episode/SGO/Index")
local FILMS = require("Module:Episode/Films/Index")
------------------------------------------------------------
-- 2. Tables globales
------------------------------------------------------------
Multi.episodes = {}
Multi.by_id = {}
Multi.by_alias = {}
Multi.by_page = {}
Multi.by_series = {}
Multi.by_season = {}
------------------------------------------------------------
-- 3. Fonction normalize() exposée
------------------------------------------------------------
local function normalize(str)
if not str then return "" end
str = str
:gsub("œ", "oe"):gsub("Œ", "oe")
:gsub("æ", "ae"):gsub("Æ", "ae")
:gsub("ç", "c"):gsub("Ç", "c")
:gsub("é", "e"):gsub("É", "e")
:gsub("è", "e"):gsub("È", "e")
:gsub("ê", "e"):gsub("Ê", "e")
:gsub("ë", "e"):gsub("Ë", "e")
:gsub("à", "a"):gsub("À", "a")
:gsub("â", "a"):gsub("Â", "a")
:gsub("ä", "a"):gsub("Ä", "a")
:gsub("î", "i"):gsub("Î", "i")
:gsub("ï", "i"):gsub("Ï", "i")
:gsub("ô", "o"):gsub("Ô", "o")
:gsub("ö", "o"):gsub("Ö", "o")
:gsub("ù", "u"):gsub("Ù", "u")
:gsub("û", "u"):gsub("Û", "u")
:gsub("ü", "u"):gsub("Ü", "u")
str = mw.ustring.lower(str)
str = mw.ustring.toNFD(str)
str = mw.ustring.gsub(str, "%pM", "")
str = str:gsub("'%s*", "")
str = str:gsub("[{}]", " ")
str = str:gsub("[!\"#%$%%&%(%)*+,%.%/%:;%<%=%>%?%@%[%]%^_%`%{%|%}%~%-]", " ")
str = str:gsub("[^%w%s]", " ")
str = str:gsub("%s+", " ")
str = mw.text.trim(str)
return str
end
Multi.normalize = normalize
------------------------------------------------------------
-- 4. Alias techniques SGX
------------------------------------------------------------
local function make_tech_aliases(series, season, episode)
if not season or not episode then
return {} -- films → aucun alias technique
end
local season_num = tonumber(season)
local e = string.format("%02d", tonumber(episode))
local s1 = string.format("%02d", season_num)
local s2 = tostring(season_num)
local s3 = tostring(season_num)
series = normalize(series)
return {
string.format("%s s%se%s", series, s1, e),
string.format("%s %sx%s", series, s2, e),
string.format("%s %s%s", series, s3, e),
}
end
------------------------------------------------------------
-- 5. Génération complète des alias
------------------------------------------------------------
local function make_aliases(ep)
local a = {}
table.insert(a, normalize(ep.page_title_fr))
table.insert(a, normalize(ep.page_title_en))
table.insert(a, normalize(ep.id))
if ep.season and ep.episode then
for _, alias in ipairs(make_tech_aliases(ep.series, ep.season, ep.episode)) do
table.insert(a, alias)
end
end
if ep.extra_aliases then
for _, alias in ipairs(ep.extra_aliases) do
table.insert(a, normalize(alias))
end
end
return a
end
------------------------------------------------------------
-- 6. Ajout d’une série complète
------------------------------------------------------------
local function append_series(series_module)
for _, ep in ipairs(series_module.episodes or {}) do
ep.aliases = make_aliases(ep)
Multi.by_id[ep.id] = ep
if ep.page_title_fr then
Multi.by_page[normalize(ep.page_title_fr)] = ep
end
if ep.page_title_en then
Multi.by_page[normalize(ep.page_title_en)] = ep
end
for _, alias in ipairs(ep.aliases) do
alias = normalize(alias)
Multi.by_alias[alias] = Multi.by_alias[alias] or {}
table.insert(Multi.by_alias[alias], ep)
end
Multi.by_series[ep.series] = Multi.by_series[ep.series] or {}
table.insert(Multi.by_series[ep.series], ep)
if ep.season then
local key = ep.series .. "-" .. ep.season
Multi.by_season[key] = Multi.by_season[key] or {}
table.insert(Multi.by_season[key], ep)
end
table.insert(Multi.episodes, ep)
end
end
------------------------------------------------------------
-- 7. Fusion des séries (UNE SEULE FOIS)
------------------------------------------------------------
append_series(SG1)
append_series(SGA)
append_series(SGU)
append_series(SGI)
append_series(SGO)
append_series(FILMS)
------------------------------------------------------------
-- 8. Enregistrer dans package.loaded (cache interne)
------------------------------------------------------------
package.loaded["Module:Episode/MultiSeries"] = Multi
function Multi.debug()
return "MultiSeries chargé OK"
end
return Multi