模組:Family tree
Family tree
This module allows for the display of English Wiktionary's hierarchical language and family ancestry data.
{{#invoke:family tree|show|language code}}
- The first parameter is the language code.
- Add
|fam=1
(alias|2=
) to show language families as separate (often redundant) nodes from their proto-languages. Ordinarily proto-languages are shown without the families that they are the parent of. - Add
|etym=1
(alias|3=
) to show all etymology languages (as children of a "parent" language or language family). Ordinarily they are not shown. - Add
|famunderproto=1
to show families on a line below the proto-languages that belong to the family and are the common ancestor of its members, and|protounderfam=1
to do the reverse. - Add
|collapsed=1
to have the top level of the tree collapsed by default.
- 原始日耳曼語 (gem-pro)
- ├───┬ 北日耳曼語支 (gmq)
- │ └───┬ 原始諾爾斯語 (gmq-pro)
- │ └───┬ 古諾爾斯語 (non)
- │ ├───┬ 古哥特蘭語 (gmq-ogt)
- │ │ └──── 哥特蘭語 (gmq-gut)
- │ ├───┬ 古東諾爾斯語 (non-oen)
- │ │ └───┬ 東斯堪地那維亞語支 (gmq-eas)
- │ │ ├───┬ 古丹麥語 (gmq-oda)
- │ │ │ └───┬ 丹麥語 (da)
- │ │ │ ├──── 旅行者丹麥語 (rmd)
- │ │ │ ├──── 日德蘭語 (jut)
- │ │ │ └──── 書面挪威語 (nb)
- │ │ │ └───┬ 丹麥語 (da)
- │ │ ├───┬ 古瑞典語 (gmq-osw)
- │ │ │ └───┬ 晚期古瑞典語 (gmq-osw-lat)
- │ │ │ └───┬ 瑞典語 (sv)
- │ │ │ └──── 斯堪地羅姆語 (rmu)
- │ │ │ └───┬ 瑞典語 (sv)
- │ │ │ └───┬ 晚期古瑞典語 (gmq-osw-lat)
- │ │ ├──── 埃爾夫達利安語 (ovd)
- │ │ ├──── 斯堪尼亞語 (gmq-scy)
- │ │ └──── 耶姆特蘭語 (gmq-jmk)
- │ │ ├───┬ 古丹麥語 (gmq-oda)
- │ │ └───┬ 東斯堪地那維亞語支 (gmq-eas)
- │ └───┬ 古西諾爾斯語 (non-own)
- │ ├──── 格陵蘭諾爾斯語 (non-grn)
- │ └───┬ 西斯堪地那維亞語支 (gmq-wes)
- │ ├───┬ 中古挪威語 (gmq-mno)
- │ │ ├───┬ 挪威語 (no)
- │ │ │ └──── 旅行者挪威語 (rmg)
- │ │ ├───┬ 新挪威語 (nn)
- │ │ │ └──── 挪威俄語 (crp-rsn)
- │ │ └──── 書面挪威語 (nb)
- │ │ ├───┬ 挪威語 (no)
- │ └───┬ 島嶼斯堪地那維亞語支 (gmq-ins)
- │ ├───┬ 中古挪威語 (gmq-mno)
- │ ├───┬ 古哥特蘭語 (gmq-ogt)
- │ └───┬ 古諾爾斯語 (non)
- │ └───┬ 原始諾爾斯語 (gmq-pro)
- ├───┬ 東日耳曼語支 (gme)
- │ ├──── 克里米亞哥特語 (gme-cgo)
- │ ├──── 古勃艮第語 (gme-bur)
- │ ├──── 哥特語 (got)
- │ └──── 汪達爾語 (xvn)
- └───┬ 西日耳曼語支 (gmw)
- └───┬ 原始西日耳曼語 (gmw-pro)
- ├───┬ 低地德語組 (gmw-lgm)
- │ └───┬ 古撒克遜語 (osx)
- │ └───┬ 中古低地德語 (gml)
- │ └───┬ 低地德語 (nds)
- │ ├───┬ 下薩克森荷蘭語 (nds-nl)
- │ │ ├──── 德倫特語 (drt)
- │ │ ├──── 斯泰靈韋夫語 (stl)
- │ │ ├──── 格羅寧根語 (gos)
- │ │ ├──── 特文特語 (twd)
- │ │ ├──── 薩蘭語 (sdz)
- │ │ ├──── 費呂沃語 (vel)
- │ │ └──── 阿赫特胡克語 (act)
- │ └───┬ 德國低地德語 (nds-de)
- │ ├──── 低地普魯士語 (nds-lpr)
- │ ├──── 威斯特法倫語 (wep)
- │ ├──── 東弗里斯蘭低地德語 (frs)
- │ └──── 門諾低地德語 (pdt)
- │ ├───┬ 下薩克森荷蘭語 (nds-nl)
- │ └───┬ 低地德語 (nds)
- │ └───┬ 中古低地德語 (gml)
- │ └───┬ 古撒克遜語 (osx)
- ├──── 低法蘭克尼亞語支 (gmw-frk)
- ├───┬ 古法蘭克語 (frk)
- │ └───┬ 古荷蘭語 (odt)
- │ ├───┬ 中古荷蘭語 (dum)
- │ │ ├──── 林堡語 (li)
- │ │ ├──── 澤蘭語 (zea)
- │ │ ├───┬ 荷蘭語 (nl)
- │ │ │ ├──── 伯比斯克里奧爾荷蘭語 (brc)
- │ │ │ ├───┬ 南非語 (af)
- │ │ │ │ └───┬ Tsotsitaal (fly)
- │ │ │ │ └──── Camtho (cmt)
- │ │ │ │ └───┬ Tsotsitaal (fly)
- │ │ │ ├──── 斯克皮克里奧爾荷蘭語 (skw)
- │ │ │ ├──── 比利時荷蘭語 (nl-BE)
- │ │ │ ├──── 澤西荷蘭語 (gmw-jdt)
- │ │ │ └──── 維京群島克里奧荷蘭語 (dcr)
- │ │ └──── 西佛蘭德語 (vls)
- │ └──── 貝格語 (gmw-bgh)
- │ ├───┬ 中古荷蘭語 (dum)
- │ └───┬ 古荷蘭語 (odt)
- ├───┬ 盎格魯-弗里斯蘭語組 (gmw-afr)
- │ ├───┬ 弗里斯蘭語組 (gmw-fri)
- │ │ └───┬ 古弗里斯蘭語 (ofs)
- │ │ ├──── 北弗里斯蘭語 (frr)
- │ │ ├──── 薩特弗里斯蘭語 (stq)
- │ │ └──── 西弗里斯蘭語 (fy)
- │ │ └───┬ 古弗里斯蘭語 (ofs)
- │ └───┬ 盎格魯語組 (gmw-ang)
- │ └───┬ 古英語 (ang)
- │ ├───┬ 中古英語 (enm)
- │ │ ├───┬ 愛爾蘭盎格魯-諾曼語組 (gmw-ian)
- │ │ │ ├──── 約拉語 (yol)
- │ │ │ └──── 芬戈語 (gmw-fin)
- │ │ └───┬ 近代英語 (en-ear)
- │ │ └───┬ 英語 (en)
- │ │ ├───┬ Maroon Spirit Language (crp-mar)
- │ │ │ └──── 克羅曼蒂語 (alv-kro)
- │ │ ├──── Ngatik Men's Creole (ngm)
- │ │ ├──── Polari (pld)
- │ │ ├──── 伯利茲克里奧爾語 (bzj)
- │ │ ├──── 利比里亞克里奧爾語 (lir)
- │ │ ├───┬ 北美英語 (en-NNN)
- │ │ │ └───┬ 美國英語 (en-US)
- │ │ │ ├──── California English (en-US-CA)
- │ │ │ └──── 加拿大英語 (en-CA)
- │ │ │ └───┬ 美國英語 (en-US)
- │ │ ├──── 印度英語 (en-IN)
- │ │ ├──── 古拉語 (gul)
- │ │ ├──── 喀麥隆皮欽語 (wes)
- │ │ ├──── 塞拉利昂克里奧爾語 (kri)
- │ │ ├──── 多巴哥克里奧爾英語 (tgh)
- │ │ ├───┬ 奧坎語 (djk)
- │ │ │ └──── Ndyuka-Trio Pidgin (njt)
- │ │ ├──── 安提瓜和巴布達克里奧爾英語 (aig)
- │ │ ├──── 尼加拉瓜克里奧爾語 (bzk)
- │ │ ├──── 尼日利亞皮欽語 (pcm)
- │ │ ├──── 島嶼克里奧爾英語 (icr)
- │ │ ├──── 巴哈馬克里奧爾語 (bah)
- │ │ ├──── 巴詹語 (bjs)
- │ │ ├───┬ 愛爾蘭英語 (en-IE)
- │ │ │ └───┬ 阿爾斯特英語 (en-uls)
- │ │ │ └──── 北愛爾蘭英語 (en-GB-NIR)
- │ │ │ └───┬ 阿爾斯特英語 (en-uls)
- │ │ ├──── 托克皮辛語 (tpi)
- │ │ ├──── 托雷斯海峽克里奧爾語 (tcs)
- │ │ ├──── 昆蒂語 (kww)
- │ │ ├──── 格瑞那達克里奧爾英語 (gcl)
- │ │ ├──── 比斯拉馬語 (bi)
- │ │ ├──── 洋涇浜英語 (cpi)
- │ │ ├───┬ 澳大利亞英語 (en-AU)
- │ │ │ └──── 澳大利亞原住民英語 (en-aae)
- │ │ ├───┬ 澳洲克里奧爾語 (rop)
- │ │ │ └──── Gurindji Kriol (gjr)
- │ │ ├──── 牙買加克里奧爾語 (jam)
- │ │ ├──── 特克斯和凱科斯群島克里奧爾英語 (tch)
- │ │ ├──── 特立尼達克里奧爾英語 (trf)
- │ │ ├──── 皮京語 (pis)
- │ │ ├──── 皮欽利斯語 (fpe)
- │ │ ├──── 皮特凱恩語 (pih)
- │ │ ├──── 盎格魯羅姆語 (rme)
- │ │ ├──── 索隆巴拉英語 (crp-slb)
- │ │ ├──── 美屬維爾京群島克里奧爾語 (vic)
- │ │ ├──── 聖文森克里奧爾語 (svc)
- │ │ ├───┬ 英國英語 (en-GB)
- │ │ │ ├──── 威爾士英語 (en-GB-WLS)
- │ │ │ ├──── 曼島英語 (en-IM)
- │ │ │ └──── 蘇格蘭英語 (en-GB-SCT)
- │ │ ├──── 蓋亞那克里奧爾英語 (gyn)
- │ │ ├──── 薩拉馬卡語 (srm)
- │ │ ├──── 薩摩亞種植園皮欽語 (crp-spp)
- │ │ ├──── 雪爾塔語 (sth)
- │ │ ├──── 非洲塞米諾爾克爾奧爾語 (afs)
- │ │ └──── 香港英語 (en-HK)
- │ │ ├───┬ Maroon Spirit Language (crp-mar)
- │ │ └───┬ 英語 (en)
- │ │ ├───┬ 愛爾蘭盎格魯-諾曼語組 (gmw-ian)
- │ ├───┬ 盎格利亞古英語 (ang-ang)
- │ │ ├───┬ 諾森布里亞古英語 (ang-nor)
- │ │ │ └───┬ 北部中古英語 (enm-nor)
- │ │ │ ├──── 喬迪英語 (en-geo)
- │ │ │ └───┬ 早期蘇格蘭語 (enm-esc)
- │ │ │ └───┬ 中古蘇格蘭語 (gmw-msc)
- │ │ │ └───┬ 低地蘇格蘭語 (sco)
- │ │ │ ├──── 北部蘇格蘭語 (sco-nor)
- │ │ │ ├──── 南部蘇格蘭語 (sco-sou)
- │ │ │ ├──── 島嶼蘇格蘭語 (sco-ins)
- │ │ │ ├──── 旅行者蘇格蘭語 (trl)
- │ │ │ └──── 阿爾斯特蘇格蘭語 (sco-uls)
- │ │ │ └───┬ 低地蘇格蘭語 (sco)
- │ │ │ └───┬ 中古蘇格蘭語 (gmw-msc)
- │ │ │ └───┬ 北部中古英語 (enm-nor)
- │ │ └──── 麥西亞古英語 (ang-mer)
- │ │ ├───┬ 諾森布里亞古英語 (ang-nor)
- │ └──── 肯特古英語 (ang-ken)
- │ ├───┬ 中古英語 (enm)
- │ └───┬ 古英語 (ang)
- │ ├───┬ 弗里斯蘭語組 (gmw-fri)
- ├──── 蘇維匯語 (gem-sue)
- └───┬ 高地德語組 (gmw-hgm)
- └───┬ 古高地德語 (goh)
- ├───┬ 中古高地德語 (gmh)
- │ ├───┬ 中東部德語 (gmw-ecg)
- │ │ ├──── 上薩克森德語 (sxu)
- │ │ └──── 西里西亞德語 (sli)
- │ ├───┬ 中部法蘭克尼亞語 (gmw-cfr)
- │ │ ├──── 漢斯立克語 (hrx)
- │ │ ├──── 特蘭西瓦尼亞薩克森語 (gmw-tsx)
- │ │ ├──── 盧森堡語 (lb)
- │ │ └──── 科隆語 (ksh)
- │ ├───┬ 巴伐利亞語 (bar)
- │ │ ├──── 戈特契語 (gmw-gts)
- │ │ ├──── 胡特利特德語 (geh)
- │ │ ├──── 辛布里語 (cim)
- │ │ └──── 默切諾語 (mhn)
- │ ├───┬ 德語 (de)
- │ │ ├───┬ 奧地利德語 (de-AT)
- │ │ │ └──── 維也納德語 (de-AT-vie)
- │ │ ├──── 拉包爾克里奧爾德語 (uln)
- │ │ ├──── 波羅的德語 (de-bal)
- │ │ └──── 瑞士德語 (de-CH)
- │ │ ├───┬ 奧地利德語 (de-AT)
- │ ├──── 意第緒語 (yi)
- │ ├──── 東法蘭克尼亞語 (vmf)
- │ ├──── 維拉莫維安語 (wym)
- │ ├───┬ 萊茵法蘭克尼亞語 (gmw-rfr)
- │ │ ├──── 伏爾加德語 (gmw-vog)
- │ │ ├──── 普法爾茨德語 (pfl)
- │ │ └──── 賓夕法尼亞德語 (pdc)
- │ ├──── 葉尼什語 (yec)
- │ ├───┬ 阿勒曼尼語 (gsw)
- │ │ ├──── Colonia Tovar German (gct)
- │ │ ├───┬ 低地阿勒曼尼語 (gsw-low)
- │ │ │ └──── 阿爾薩斯阿勒曼尼語 (gsw-FR-als)
- │ │ ├───┬ 施瓦本語 (swg)
- │ │ │ └──── 薩圖馬雷施瓦本語 (gmw-stm)
- │ │ ├───┬ 至高阿勒曼尼語 (gsw-hst)
- │ │ │ └──── 瓦爾瑟德語 (wae)
- │ │ └──── 高地阿勒曼尼語 (gsw-hig)
- │ └──── 齊普澤德語 (gmw-zps)
- │ ├───┬ 中東部德語 (gmw-ecg)
- └──── 古倫巴底語 (lng)
- ├───┬ 中古高地德語 (gmh)
- └───┬ 古高地德語 (goh)
- ├───┬ 低地德語組 (gmw-lgm)
- └───┬ 原始西日耳曼語 (gmw-pro)
- ├───┬ 北日耳曼語支 (gmq)
Trees
The template include limit prevents this page from showing all of Wiktionary's current language trees. Above is a sample language tree. When making changes to the Module:languages data, you may need to refresh
this page to see an update.
See also
--[=[
Authors: [[User:kc_kennylau]], [[User:JohnC5]], [[User:Erutuon]], [[User:Suzukaze-c]], [[User:Theknightwho]]
--]=]
local export = {}
local regular_languages = require("Module:languages/code to canonical name")
local etymology_languages = require("Module:etymology languages/code to canonical name")
local families = require("Module:families/code to canonical name")
function export.find_subtree(t, code)
for _, val in ipairs(t) do
if val.name == code then -- "name" is really code
return {val}
end
local result = export.find_subtree(val, code)
if result ~= nil then
return result
end
end
end
local family_icon = "F"
local etymology_language_icon = "E"
local proto_language_icon = family_icon
local family_with_proto_language_icon = family_icon
local function format_node(code, is_protolanguage_or_has_protolanguage)
local canonical_name, category_name, class, icon, tooltip
if regular_languages[code] then
canonical_name = regular_languages[code]
category_name = canonical_name
class = "familytree-lang"
if is_protolanguage_or_has_protolanguage then
class = class .. ' familytree-protolang'
icon = proto_language_icon
end
elseif etymology_languages[code] then
canonical_name = etymology_languages[code]
class = "familytree-etymlang"
icon = etymology_language_icon
tooltip = "Etymology language"
elseif families[code] then
canonical_name = families[code]
category_name = canonical_name
class = "familytree-family"
if is_protolanguage_or_has_protolanguage then
class = class .. ' familytree-hasprotolang'
icon = family_with_proto_language_icon
else
icon = family_icon
end
tooltip = "Language family"
end
return '<span class="' .. class .. '" '
.. (tooltip and 'title="' .. tooltip .. '"' or '') .. '>'
.. '[[:Category:' .. (category_name or canonical_name) .. '|'
.. canonical_name
.. ' <span class="familytree-code">(' .. code .. ')</span>]]'
.. (icon and ' <span class="familytree-icon">' .. icon .. '</span>' or '')
.. '</span>'
end
-- If neither options.show_all_families or options.show_etymology_languages is
-- falsy, then this function does nothing.
local function filter_nested_data(nested_data, options, protolanguage_of, is_protolanguage)
if not nested_data then -- ???
return nil
else
local name = nested_data.name
local first_child = nested_data[1]
-- This indicates that new_nested_data below should only be returned
-- if it contains non-etymology languages.
local check_for_non_etymology_children = false
-- If `show_all_families` is false and this is a family and its only
-- child is its proto-language, then replace the family with the
-- proto-language.
if options.hide_families_with_protolanguages and name and families[name]
and first_child and not nested_data[2]
and protolanguage_of[name] == first_child.name then
is_protolanguage[first_child.name] = true
return filter_nested_data(first_child, options, protolanguage_of, is_protolanguage)
elseif options.hide_etymology_languages
and etymology_languages[name] then
if nested_data[1] then
check_for_non_etymology_children = true
else
return nil
end
end
local new_nested_data = { name = name }
local i = 0
for _, subtable in ipairs(nested_data) do
subtable = filter_nested_data(subtable, options, protolanguage_of, is_protolanguage)
if subtable then
i = i + 1
new_nested_data[i] = subtable
end
end
if not check_for_non_etymology_children or new_nested_data[1] then
return new_nested_data
end
end
end
local function make_node(code, is_protolanguage, protolanguage_of)
return '</span> ' .. format_node(code,
is_protolanguage[code] or protolanguage_of[code] ~= nil)
end
local function only_child_is_protolanguage(tree, options, protolanguage_of)
return (options.family_under_protolanguage
or options.protolanguage_under_family)
and tree[1] and protolanguage_of[tree.name] == tree[1].name
end
export.are_all_children_etymology_languages = require("Module:memoize")(function (nested_data)
if not nested_data[1] then
return nil
end
for _, child in ipairs(nested_data) do
if not etymology_languages[child.name]
or export.are_all_children_etymology_languages(child) == false then
return false
end
end
return true
end)
local customcollapsible_number = 0
local customcollapsible_prefix = "familytree"
local function get_customcollapsible_id()
customcollapsible_number = customcollapsible_number + 1
return customcollapsible_prefix .. customcollapsible_number
end
local no_break_space = mw.ustring.char(0xA0)
local level_separator = (no_break_space):rep(3)
local expandtext, collapsetext = "[+]─", "[-]┬"
local function make_tree(data, is_protolanguage, protolanguage_of, options, prefix)
local result = {}
local function ins(val)
table.insert(result, val)
end
-- This tag is closed in the node generated by make_node.
prefix = prefix or '<span class="familytree-linedrawing">'
local branch = "├"
local next_level = prefix .. "│" .. level_separator
local length = #data
for i, val in ipairs(data) do
if i == length then
branch = "└"
next_level = prefix .. level_separator .. no_break_space
end
local code = val.name
local language_or_family_node =
make_node(code, is_protolanguage, protolanguage_of)
if not val[1] then
ins('<li>' .. prefix .. branch .. options.sterile_branch_text
.. language_or_family_node .. '</li>')
else
local customcollapsible_id = get_customcollapsible_id()
ins('<li>' .. prefix .. branch
.. '<span class="familytree-toggle mw-customtoggle-'
.. customcollapsible_id .. '">───┬</span>')
-- name me!
local flag = (options.family_under_protolanguage
or options.protolanguage_under_family)
and only_child_is_protolanguage(val, options, protolanguage_of)
local top_node
if flag then
code = val[1].name
val = val[1]
top_node = make_node(code, is_protolanguage, protolanguage_of)
if options.protolanguage_under_family then
top_node, language_or_family_node =
language_or_family_node, top_node
end
end
local all_children_are_etymology_languages =
export.are_all_children_etymology_languages(val)
local collapsible_ul = '<ul class="mw-collapsible'
.. (all_children_are_etymology_languages
and ' familytree-only-etym-children'
or '') .. '" '
.. 'id="mw-customcollapsible-' .. customcollapsible_id
.. '" data-expandtext="' .. expandtext
.. '" data-collapsetext="' .. collapsetext .. '">'
if flag then
ins(top_node
.. collapsible_ul .. '<li>' .. prefix
.. (i == length and no_break_space or "│")
.. level_separator .. "│")
end
ins(language_or_family_node)
if not flag then
ins(collapsible_ul)
end
-- Can't get default collapsibility script to apply the data-expandtext
-- and data-collapsetext attribute values to the custom toggle,
-- so have to have a custom script do it.
ins(make_tree(val, is_protolanguage, protolanguage_of, options, next_level))
ins('</ul></li>')
end
end
return table.concat(result)
end
local function get_number_parameter_in_range(args, arg, low, high)
local val = args[arg]
if val == "" or val == nil then
val = nil
else
val = tonumber(val)
if not (type(val) == "number"
and 0 <= val and val <= 6) then
error("Expected nothing or number between " .. low .. " and "
.. high .. " in parameter |" .. arg .. "=.")
end
end
return val
end
function export.show(frame)
local args = frame.args
local descendants_of = args[1]
local to_boolean = require("Module:yesno")
-- Determines whether families that have proto-languages will be shown.
local show_all_families = to_boolean(args[2] or args.fam)
-- Determines whether all etymology languages will be shown.
local show_etymology_languages = to_boolean(args[3] or args.etym)
-- help! parameter name too long!
local sterile_branch_length = get_number_parameter_in_range(args, "sterile_branch_length", 0, 6)
-- Determines whether (if all families are shown) a family will be shown
-- on a line directly under and at the same level as its proto-language,
-- or the proto-language on a line directly under and at the same level as
-- its family.
local family_under_protolanguage = to_boolean(args.famunderproto)
local protolanguage_under_family = to_boolean(args.protounderfam)
if family_under_protolanguage and protolanguage_under_family then
error("Kindly choose between proto-language under family and family under proto-language.")
end
return export.print_children(descendants_of, {
hide_families_with_protolanguages = not show_all_families,
hide_etymology_languages = not show_etymology_languages,
family_under_protolanguage = family_under_protolanguage,
protolanguage_under_family = protolanguage_under_family,
sterile_branch_length = sterile_branch_length,
collapsed = require("Module:yesno")(args.collapsed)
})
end
function export.print_children(descendants_of, options)
local m_languages = require("Module:languages")
local m_table = require("Module:table")
local make_auto_subtabler = require("Module:auto-subtable")
descendants_of = m_languages.getByCode(descendants_of, nil, true, true)
local names = {}
local protolanguage_of = {}
local children = make_auto_subtabler{}
local descendants = descendants_of:getDescendantCodes()
table.insert(descendants, descendants_of:getCode())
if descendants_of:hasType("family") then
protolanguage_of[descendants_of:getCode()] = descendants_of:getProtoLanguageCode()
end
local memoized = {}
local get = function(code, func, ...)
local ret = memoized[code] or func(...)
if code then
memoized[code] = ret
end
return ret
end
for _, descendant_code in ipairs(descendants) do
-- Inner "repeat until true" loop allows break to work like continue, as it will always only run once.
repeat
local descendant = get(descendant_code, m_languages.getByCode, descendant_code, nil, true, true)
names[descendant_code] = descendant:getCanonicalName():gsub("Proto%-", "")
if descendant:hasType("language") then
local ancestors = m_table.shallowCopy(descendant:getAncestorCodes())
local parent_code = descendant:getParentCode()
if parent_code and descendant:hasType("etymology-only") then
local parent = get(parent_code, descendant.getParent, descendant)
if m_table.deepEquals(parent:getAncestorCodes(), ancestors) and
descendant:getFamilyCode() == parent:getFamilyCode() then
table.insert(children[parent:getCode()], descendant_code)
break
end
end
if #ancestors > 0 then
for _, ancestor in ipairs(ancestors) do
table.insert(children[ancestor], descendant_code)
end
break
end
else
local protolang = descendant:getProtoLanguageCode()
protolanguage_of[descendant_code] = protolang
if protolang and descendant:hasAncestor(protolang) then
table.insert(children[protolang], descendant_code)
break
end
end
local family_code = descendant:getFamilyCode()
if family_code then
local family = get(family_code, descendant.getFamily, descendant)
local protolang = get(family:getProtoLanguageCode(), family.getProtoLanguage, family)
if not protolanguage_of[family] then
protolanguage_of[family] = protolang and protolang:getCode()
end
if protolang and protolang:inFamily(family) and protolang:getCode() ~= descendant_code then
table.insert(children[protolang:getCode()], descendant_code)
else
table.insert(children[family:getCode()], descendant_code)
end
end
until true
end
-- No more auto subtabling needed.
children = children:un_auto_subtable()
-- Copy to new table, to filter out unwanted ancestors from descendants with multiple ancestors, where some are not descendants of the target language.
local parent_to_children_map = {}
for _, code in ipairs(descendants) do
parent_to_children_map[code] = children[code]
end
local function make_nested(data, children)
local make_nil = {}
for key, val in pairs(data) do
if type(key) == "number" then
if children[val] then
data[val] = make_nested(children[val], children)
table.insert(make_nil, key)
end
else
data[key] = make_nested(val, children)
end
end
if make_nil[2] then -- Make sure larger keys are removed first.
table.sort(make_nil, function (a, b) return a > b end)
end
for _, key in ipairs(make_nil) do
table.remove(data, key)
end
return data
end
local nested = make_nested(parent_to_children_map, parent_to_children_map)
local function deep_sort(current)
local result = {}
local is_table = {}
for key, val in pairs(current) do
if type(key) == "number" then
table.insert(result, val)
else
is_table[key] = true
table.insert(result, key)
end
end
table.sort(result, function(code1, code2)
return names[code1] < names[code2]
end)
for i = 1, #result do
if is_table[result[i]] then
local name = result[i]
result[i] = deep_sort(current[result[i]])
result[i].name = name
else
result[i] = { name = result[i] }
end
end
return result
end
nested = deep_sort(nested)
data = { nested = nested, protolanguage_of = protolanguage_of }
local nested_data, protolanguage_of = data.nested, data.protolanguage_of
nested_data = export.find_subtree(nested_data, descendants_of:getCode())
-- Return nil instead of a tree with only the root node.
if options.must_have_descendants and (nested_data == nil or #nested_data == 0 or nested_data[1] and #nested_data[1] == 0) then
return nil
end
local is_protolanguage = {}
if options.hide_families_with_protolanguages or options.hide_etymology_languages then
nested_data = filter_nested_data(nested_data, {
hide_families_with_protolanguages = options.hide_families_with_protolanguages,
hide_etymology_languages = options.hide_etymology_languages,
}, protolanguage_of, is_protolanguage)
end
if not nested_data or not next(nested_data) then
return nil
end
local result = {'<div class="familytree"><ul>'}
local function ins(val)
table.insert(result, val)
end
local tree_options = {
sterile_branch_text = '<span class="familytree-branch">'
.. ("─"):rep(options.sterile_branch_length or 4)
.. '</span>',
family_under_protolanguage = options.family_under_protolanguage,
protolanguage_under_family = options.protolanguage_under_family,
}
local collapsetext, expandtext = 'Collapse', 'Expand'
for i, subtable in ipairs(nested_data) do
-- top language name
ins('<li>')
-- name me!
local flag = (options.family_under_protolanguage
or options.protolanguage_under_family)
and only_child_is_protolanguage(subtable, options, protolanguage_of)
local top_node = format_node(subtable.name)
local next_node
if flag then
subtable = subtable[1]
next_node = format_node(subtable.name)
if options.family_under_protolanguage then
top_node, next_node = next_node, top_node
end
end
ins(top_node)
-- top toggle
local customcollapsible_id = get_customcollapsible_id()
ins('<span class="familytree-toptoggle mw-customtoggle-'
.. customcollapsible_id .. '" style="display: none;">')
ins(options.collapsed and expandtext or collapsetext)
ins('</span>')
if flag then
ins('<li>')
ins(next_node)
end
-- tree
ins('<ul class="mw-collapsible')
if options.collapsed then
ins(' mw-collapsed')
end
ins('" id="mw-customcollapsible-' .. customcollapsible_id)
ins('" data-expandtext="' .. expandtext)
ins('" data-collapsetext="' .. collapsetext .. '">')
ins(make_tree(subtable, is_protolanguage, protolanguage_of, tree_options))
if flag then
ins('</li>')
end
ins('</ul></li>')
end
ins('</ul></div>')
ins(require("Module:TemplateStyles")("Module:family tree/style.css"))
return table.concat(result)
end
return export