ഘടകം:Test001

വിക്കിപീഡിയ, ഒരു സ്വതന്ത്ര വിജ്ഞാനകോശം.

ഈ ഘടകത്തിന്റെ വിവരണം ഘടകം:Test001/വിവരണം എന്ന താളിൽ നിർമ്മിക്കാവുന്നതാണ്

--local p = {}
--function p.hello(frame)
--    return 'Hello'
--end
--return p

-- Module for converting between different representations of numbers. See talk page for user documentation.
-- For unit tests see: [[Module:ConvertNumeric/tests]]
 
local ones_position = {
    [0] = 'പൂജ്യം',
    [1] = 'ഒന്നു്',
    [2] = 'രണ്ട്',
    [3] = 'മൂന്നു്',
    [4] = 'four',
    [5] = 'five',
    [6] = 'six',
    [7] = 'seven',
    [8] = 'eight',
    [9] = 'nine',
    [10] = 'ten',
    [11] = 'പതിനൊന്നു്',
    [12] = 'twelve',
    [13] = 'thirteen',
    [14] = 'fourteen',
    [15] = 'fifteen',
    [16] = 'sixteen',
    [17] = 'seventeen',
    [18] = 'eighteen',
    [19] = 'nineteen'
}
 
local ones_position_ord = {
    [0] = 'പൂജ്യാമതു്',
    [1] = 'ഒന്നാമതു്',
    [2] = 'രണ്ടാം',
    [3] = 'മൂന്നാമത്തെ',
    [4] = 'fourth',
    [5] = 'fifth',
    [6] = 'sixth',
    [7] = 'seventh',
    [8] = 'eighth',
    [9] = 'ninth',
    [10] = 'tenth',
    [11] = 'eleventh',
    [12] = 'twelfth',
    [13] = 'thirteenth',
    [14] = 'fourteenth',
    [15] = 'fifteenth',
    [16] = 'sixteenth',
    [17] = 'seventeenth',
    [18] = 'eighteenth',
    [19] = 'nineteenth'
}
 
local ones_position_plural = {
    [0] = 'zeros',
    [1] = 'ones',
    [2] = 'twos',
    [3] = 'threes',
    [4] = 'fours',
    [5] = 'fives',
    [6] = 'sixes',
    [7] = 'sevens',
    [8] = 'eights',
    [9] = 'nines',
    [10] = 'tens',
    [11] = 'elevens',
    [12] = 'twelves',
    [13] = 'thirteens',
    [14] = 'fourteens',
    [15] = 'fifteens',
    [16] = 'sixteens',
    [17] = 'seventeens',
    [18] = 'eighteens',
    [19] = 'nineteens'
}
 
local tens_position = {
    [2] = 'twenty',
    [3] = 'thirty',
    [4] = 'forty',
    [5] = 'fifty',
    [6] = 'sixty',
    [7] = 'seventy',
    [8] = 'eighty',
    [9] = 'ninety'
}
 
local tens_position_ord = {
    [2] = 'twentieth',
    [3] = 'thirtieth',
    [4] = 'fortieth',
    [5] = 'fiftieth',
    [6] = 'sixtieth',
    [7] = 'seventieth',
    [8] = 'eightieth',
    [9] = 'ninetieth'
}
 
local tens_position_plural = {
    [2] = 'twenties',
    [3] = 'thirties',
    [4] = 'forties',
    [5] = 'fifties',
    [6] = 'sixties',
    [7] = 'seventies',
    [8] = 'eighties',
    [9] = 'തൊണ്ണൂറുകൾ'
}
 
local groups = {
    [1] = 'thousand',
    [2] = 'million',
    [3] = 'billion',
    [4] = 'trillion',
    [5] = 'quadrillion',
    [6] = 'quintillion',
    [7] = 'sextillion',
    [8] = 'septillion',
    [9] = 'octillion',
    [10] = 'nonillion',
    [11] = 'decillion',
    [12] = 'undecillion',
    [13] = 'duodecillion',
    [14] = 'tredecillion',
    [15] = 'quattuordecillion',
    [16] = 'quindecillion',
    [17] = 'sexdecillion',
    [18] = 'septendecillion',
    [19] = 'octodecillion',
    [20] = 'novemdecillion',
    [21] = 'vigintillion',
    [22] = 'unvigintillion',
    [23] = 'duovigintillion',
    [24] = 'tresvigintillion',
    [25] = 'quattuorvigintillion',
    [26] = 'quinquavigintillion',
    [27] = 'sesvigintillion',
    [28] = 'septemvigintillion',
    [29] = 'octovigintillion',
    [30] = 'novemvigintillion',
    [31] = 'trigintillion',
    [32] = 'untrigintillion',
    [33] = 'duotrigintillion',
    [34] = 'trestrigintillion',
    [35] = 'quattuortrigintillion',
    [36] = 'quinquatrigintillion',
    [37] = 'sestrigintillion',
    [38] = 'septentrigintillion',
    [39] = 'octotrigintillion',
    [40] = 'noventrigintillion',
    [41] = 'quadragintillion',
    [51] = 'quinquagintillion',
    [61] = 'sexagintillion',
    [71] = 'septuagintillion',
    [81] = 'octogintillion',
    [91] = 'nonagintillion',
    [101] = 'centillion',
    [102] = 'uncentillion',
    [103] = 'duocentillion',
    [104] = 'trescentillion',
    [111] = 'decicentillion',
    [112] = 'undecicentillion',
    [121] = 'viginticentillion',
    [122] = 'unviginticentillion',
    [131] = 'trigintacentillion',
    [141] = 'quadragintacentillion',
    [151] = 'quinquagintacentillion',
    [161] = 'sexagintacentillion',
    [171] = 'septuagintacentillion',
    [181] = 'octogintacentillion',
    [191] = 'nonagintacentillion',
    [201] = 'ducentillion',
    [301] = 'trecentillion',
    [401] = 'quadringentillion',
    [501] = 'quingentillion',
    [601] = 'sescentillion',
    [701] = 'septingentillion',
    [801] = 'octingentillion',
    [901] = 'nongentillion',
    [1001] = 'millinillion',
}
 
roman_numerals = {
    I = 1,
    V = 5,
    X = 10,
    L = 50,
    C = 100,
    D = 500,
    M = 1000
}
 
-- Converts a given valid roman numeral (and some invalid roman numerals) to a number. Returns -1, errorstring on error
function roman_to_numeral(roman)
    if type(roman) ~= "string" then return -1, "roman numeral not a string" end
    local rev = roman:reverse()
    local raising = true
    local last = 0
    local result = 0
    for i = 1, #rev do
        local c = rev:sub(i, i)
        local next = roman_numerals[c]
        if next == nil then return -1, "roman numeral contains illegal character " .. c end
        if next > last then
            result = result + next
            raising = true
        elseif next < last then
            result = result - next
            raising = false
        elseif raising then
            result = result + next
        else
            result = result - next
        end
        last = next
    end
    return result
end
 
-- Converts a given integer between 0 and 100 to English text (e.g. 47 -> forty-seven)
function numeral_to_english_less_100(num, ordinal, plural, zero)
    local terminal_ones, terminal_tens
    if ordinal then
        terminal_ones = ones_position_ord
        terminal_tens = tens_position_ord
    elseif plural then
        terminal_ones = ones_position_plural
        terminal_tens = tens_position_plural
    else
        terminal_ones = ones_position
        terminal_tens = tens_position
    end
 
    if num == 0 and zero ~= nil then
        return zero
    elseif num < 20 then
        return terminal_ones[num]
    elseif num % 10 == 0 then
        return terminal_tens[num / 10]
    else
        return tens_position[math.floor(num / 10)] .. '-' .. terminal_ones[num % 10]
    end
end
 
function standard_suffix(ordinal, plural)
    if ordinal then return 'th' end
    if plural then return 's' end
    return ''
end
 
-- Converts a given integer (in string form) between 0 and 1000 to English text (e.g. 47 -> forty-seven)
function numeral_to_english_less_1000(num, use_and, ordinal, plural, zero)
    num = tonumber(num)
    if num < 100 then
        return numeral_to_english_less_100(num, ordinal, plural, zero)
    elseif num % 100 == 0 then
        return ones_position[num/100] .. ' hundred' .. standard_suffix(ordinal, plural)
    else
        return ones_position[math.floor(num/100)] .. ' hundred ' .. (use_and and 'and ' or '') .. numeral_to_english_less_100(num % 100, ordinal, plural, zero)
    end
end
 
-- Converts a number expressed as a string in scientific notation to a string in standard decimal notation
-- e.g. 1.23E5 -> 123000, 1.23E-5 = .0000123. Conversion is exact, no rounding is performed.
function scientific_notation_to_decimal(num)
    local exponent, subs = num:gsub("^%-?%d*%.?%d*%-?[Ee]([+%-]?%d+)$", "%1")
    if subs == 0 then return num end  -- Input not in scientific notation, just return unmodified
    exponent = tonumber(exponent)
 
    local negative = num:find("^%-")
    _, decimal_pos = num:find("%.")
    -- Mantissa will consist of all decimal digits with no decimal point
    local mantissa = num:gsub("^%-?(%d*)%.?(%d*)%-?[Ee][+%-]?%d+$", "%1%2")
    if negative and decimal_pos then decimal_pos = decimal_pos - 1 end
    if not decimal_pos then decimal_pos = #mantissa + 1 end
    local prev_len = #num
 
    -- Remove leading zeros unless decimal point is in first position
    while decimal_pos > 1 and mantissa:sub(1,1) == '0' do
        mantissa = mantissa:sub(2)
        decimal_pos = decimal_pos - 1
    end
    -- Shift decimal point right for exponent > 0
    while exponent > 0 do
        decimal_pos = decimal_pos + 1
        exponent = exponent - 1
        if decimal_pos > #mantissa + 1 then mantissa = mantissa .. '0' end
        -- Remove leading zeros unless decimal point is in first position
        while decimal_pos > 1 and mantissa:sub(1,1) == '0' do
            mantissa = mantissa:sub(2)
            decimal_pos = decimal_pos - 1
        end
    end
    -- Shift decimal point left for exponent < 0
    while exponent < 0 do
        if decimal_pos == 1 then
            mantissa = '0' .. mantissa
        else
            decimal_pos = decimal_pos - 1
        end
        exponent = exponent + 1
    end
 
    -- Insert decimal point in correct position and return
    return (negative and '-' or '') .. mantissa:sub(1, decimal_pos - 1) .. '.' .. mantissa:sub(decimal_pos)
end
 
-- Rounds a number to the nearest integer
function round_num(x)
    if x%1 >= 0.5 then
        return math.ceil(x)
    else 
        return math.floor(x)
    end
end
 
-- Rounds a number to the nearest two-word number (round = up, down, or "on" for round to nearest)
-- Numbers with two digits before the decimal will be rounded to an integer as specified by round.
-- Larger numbers will be rounded to a number with only one nonzero digit in front and all other digits zero.
-- Negative sign is preserved and does not count towards word limit.
function round_for_english(num, round)
    -- If an integer with at most two digits, just return
    if num:find("^%-?%d?%d%.?$") then return num end
 
    local negative = num:find("^%-")
    if negative then
        -- We're rounding magnitude so flip it
        if round == 'up' then round = 'down' elseif round == 'down' then round = 'up' end
    end
 
    -- If at most two digits before decimal, round to integer and return
    local _, _, small_int, trailing_digits, round_digit = num:find("^%-?(%d?%d?)%.((%d)%d*)$")
    if small_int then
        local small_int_len = #small_int
        if small_int == '' then small_int = '0' end
        if (round == 'up' and trailing_digits:find('[1-9]')) or (round == 'on' and tonumber(round_digit) >= 5) then
            small_int = tostring(tonumber(small_int) + 1)
        end
        return (negative and '-' or '') .. small_int
    end
 
    -- When rounding up, any number with > 1 nonzero digit will round up (e.g. 1000000.001 rounds up to 2000000)
    local nonzero_digits = 0
    for digit in num:gfind("[1-9]") do
        nonzero_digits = nonzero_digits + 1
    end
 
    num = num:gsub("%.%d*$", "") -- Remove decimal part
    -- Second digit used to determine which way to round lead digit
    local _, _, lead_digit, round_digit, round_digit_2, rest = num:find("^%-?(%d)(%d)(%d)(%d*)$")
    if tonumber(lead_digit .. round_digit) < 20 and (1 + #rest) % 3 == 0 then
        -- In English numbers < 20 are one word so put 2 digits in lead and round based on 3rd
        lead_digit = lead_digit .. round_digit
        round_digit = round_digit_2
    else
        rest = round_digit_2 .. rest
    end
 
    if (round == 'up' and nonzero_digits > 1) or (round == 'on' and tonumber(round_digit) >= 5) then
        lead_digit = tostring(tonumber(lead_digit) + 1)
    end
    -- All digits but lead digit will turn to zero
    rest = rest:gsub("%d", "0")
    return (negative and '-' or '') .. lead_digit .. '0' .. rest
end
 
-- Takes a decimal number and converts it to English text.
-- num (string): the number to convert. Can be arbitrarily large decimal, such as "-123456789123456789.345",
--               and can use scientific notation (e.g. "1.23E5"). May fail for very large numbers not listed in "groups" such as "1E4000".
-- capitalize (boolean): whether to capitalize the result (e.g. 'One' instead of 'one')
-- use_and (boolean): whether to use the word 'and' between tens/ones place and higher places
-- hyphenate (boolean): whether to hyphenate all words in the result, useful for use as an adjective
-- ordinal (boolean): whether to produce an ordinal (e.g. 'first' instead of 'one')
-- plural (boolean): whether to pluralize the resulting number
-- links: nil: do not add any links; 'on': link "billion" and larger to Orders of magnitude article;
--        any other text: list of numbers to link (e.g. billion,quadrillion)
-- negative_word: Word to use for negative sign (typically 'negative' or 'minus')
-- round: nil or '': no rounding; 'on': round to nearest two-word number; 'up'/'down': round up/down to two-word number
function _numeral_to_english(num, capitalize, use_and, hyphenate, ordinal, plural, links, negative_word, round, zero)
    num = scientific_notation_to_decimal(num)
 
    if round and round ~= '' and round ~= 'on' and round ~= 'up' and round ~= 'down' then return 'Invalid rounding mode' end
    if round and round ~= '' then num = round_for_english(num, round) end
 
    -- Separate into negative sign, num (digits before decimal), decimal_places (digits after decimal)
    local negative = num:find("^%-")
    local decimal_places, subs = num:gsub("^%-?%d*%.(%d+)$", "%1")
    if subs == 0 then decimal_places = nil end
    num, subs = num:gsub("^%-?(%d*)%.?%d*$", "%1")
    if num == '' and decimal_places then num = '0' end
    if subs == 0 or num == '' then return 'Invalid decimal numeral' end
 
    -- For each group of 3 digits except the last one, print with appropriate group name (e.g. million)
    local s = ''
    while #num > 3 do
        if s ~= '' then s = s .. ' ' end
        local group_num = math.floor((#num - 1) / 3)
        local group = groups[group_num]
        local group_digits = #num - group_num*3
        s = s .. numeral_to_english_less_1000(num:sub(1, group_digits), false, false, false, zero) .. ' '
        if ((links == 'on' and group_num >= 3) or links:find(group)) and group_num <= 13 then
            s = s .. '[[Orders_of_magnitude_(numbers)#10' .. group_num*3 .. '|' .. group .. ']]'
        else
            s = s .. group
        end
        num = num:sub(1 + group_digits)
        num = num:gsub("^0*", "")   -- Trim leading zeros
    end
 
    -- Handle final three digits of integer part
 
    if s ~= '' and num ~= '' then
        if #num > 2 or omit_and then
            s = s .. ' '
        else
            s = s .. ' and '
        end
    end
    if s == '' or num ~= '' then
        s = s .. numeral_to_english_less_1000(num, use_and, ordinal, plural, zero)
    elseif ordinal or plural then
        -- Round numbers like "one million" take standard suffixes for ordinal/plural
        s = s .. standard_suffix(ordinal, plural)
    end
 
    -- For decimal places (if any) output "point" followed by spelling out digit by digit
    if decimal_places then
        s = s .. ' point'
        for i = 1, #decimal_places do
            s = s .. ' ' .. ones_position[tonumber(decimal_places:sub(i,i))]
        end
    end
 
    s = s:gsub("^%s*(.-)%s*$", "%1")   -- Trim whitespace
    if ordinal and plural then s = s .. 's' end  -- s suffix works for all ordinals
    if negative and s ~= zero then s = negative_word .. ' ' .. s end
    s = s:gsub("negative zero", "zero")
    if hyphenate then s = s:gsub("%s", "-") end
    if capitalize then s = s:gsub("^%l", string.upper) end
 
    return s
end
 
local p = {}
 
function p.numeral_to_english(frame)
    local num = frame.args[1]
    num = num:gsub("^%s*(.-)%s*$", "%1")   -- Trim whitespace
    num = num:gsub(",", "")   -- Remove commas
    if not num:find("^%-?%d*%.?%d*%-?[Ee]?[+%-]?%d*$") then
        -- Input not in a valid format, try to pass it through #expr to see if that produces a number
        -- (e.g. "3 + 5" will become 8)
        num = frame:preprocess('{{#expr: ' .. num .. '}}')
    end
 
    -- Convert all args passed through frame into normal arguments to helper function
    local case = frame.args['case']
    local sp = frame.args['sp']
    local adj = frame.args['adj']
    local ord = frame.args['ord']
    local pl = frame.args['pl']
    local lk = frame.args['lk'] or ''
    local negative = frame.args['negative'] or 'negative'
    local round = frame.args['round'] or nil    
    local zero = frame.args['zero'] or nil
    return _numeral_to_english(num, case == 'U' or case == 'u', sp ~= 'us', adj == 'on', ord == 'on', pl == 'on', lk, negative, round, zero)
end
 
---- recursive function for p.decToHex
function decToHexDigit(dec)
   local dig = {"0","1","2","3","4","5","6","7","8","9","A","B","C","D","E","F"}
   local div = math.floor(dec/16)
   local mod = dec-(16*div)
   if div >= 1 then return decToHexDigit(div)..dig[mod+1] else return dig[mod+1] end
end -- I think this is supposed to be done with a tail call but first I want something that works at all
---- finds all the decimal numbers in the input text and hexes each of them
function p.decToHex(frame)
   local args=frame.args
   local parent=frame.getParent(frame)
   local pargs={}
   if parent then pargs=parent.args end
   local text=args[1] or pargs[1] or ""
   local minlength=args.minlength or pargs.minlength or 1
   minlength=tonumber(minlength)
   prowl=mw.ustring.gmatch(text,"(.-)(%d+)")
   local output=""
   repeat
      local chaff,dec=prowl()
      if not(dec) then break end
      local hex=decToHexDigit(dec)
      while (mw.ustring.len(hex)<minlength) do hex="0"..hex end
      output=output..chaff..hex
   until false
   local chaff=mw.ustring.match(text,"(%D+)$") or ""
   return output..chaff
end
 
return p
"https://ml.wikipedia.org/w/index.php?title=ഘടകം:Test001&oldid=1771594" എന്ന താളിൽനിന്ന് ശേഖരിച്ചത്