Difference between revisions of "Ftf snippets"

From GiderosMobile
 
(3 intermediate revisions by the same user not shown)
Line 18: Line 18:
 
</syntaxhighlight>
 
</syntaxhighlight>
  
'''Example 2''': assume you have a boolean called ''b'' and you want to set its default value to false
+
'''Example 2''': assume you have a boolean called ''b'' and you want to set its default value to '''false'''
 
<syntaxhighlight lang="lua">
 
<syntaxhighlight lang="lua">
 
-- in lua you do
 
-- in lua you do
Line 26: Line 26:
 
</syntaxhighlight>
 
</syntaxhighlight>
  
'''Example 3''': assume you have a boolean called ''b'' and you want to set its default value to true
+
'''Example 3''': assume you have a boolean called ''b'' and you want to set its default value to '''true'''
 
<syntaxhighlight lang="lua">
 
<syntaxhighlight lang="lua">
 
-- in lua you do
 
-- in lua you do
Line 1,266: Line 1,266:
 
g = nil -- cleanup
 
g = nil -- cleanup
 
-- ...
 
-- ...
 +
end
 +
</syntaxhighlight>
 +
 +
=== Calculate Angle between 2 Points '''@GregBUG''' ===
 +
<syntaxhighlight lang="lua">
 +
local qAtan2 = math.atan2
 +
 +
local function calcAngle(xA, yA, xB, yB)
 +
return qAtan2((yB - yA), (xB - xA))
 +
end
 +
</syntaxhighlight>
 +
 +
=== Calculate Distance between 2 Points '''@GregBUG''' ===
 +
<syntaxhighlight lang="lua">
 +
local function distance(xA, yA, xB, yB)
 +
local dx = xB - xA
 +
local dy = yB - yA
 +
-- return (dx*dx + dy*dy)^0.5 -- much faster than math.sqrt
 +
return (dx^2 + dy^2)^0.5 -- this is an even faster one, and the fastest you can get!
 
end
 
end
 
</syntaxhighlight>
 
</syntaxhighlight>
Line 1,378: Line 1,397:
 
end
 
end
 
</syntaxhighlight>
 
</syntaxhighlight>
 
== DEBUGGING ==
 
 
=== TEXTURE MEMORY USAGE '''@SinisterSoft''' ===
 
<syntaxhighlight lang="lua">
 
local temp, oldusage
 
temp=(application:getTextureMemoryUsage()/1024)
 
if temp~=oldusage then
 
oldusage=temp
 
print("* 1 Texture usage: "..temp.." MB")
 
end
 
</syntaxhighlight>
 
 
== ALGOS ==
 
  
 
=== DISPLAY LIST AS GRID ALGO '''@sslivka''' ===
 
=== DISPLAY LIST AS GRID ALGO '''@sslivka''' ===
Line 1,432: Line 1,437:
 
end
 
end
 
</syntaxhighlight>
 
</syntaxhighlight>
 +
 +
== DEBUGGING ==
 +
 +
=== TEXTURE MEMORY USAGE '''@SinisterSoft''' ===
 +
<syntaxhighlight lang="lua">
 +
local temp, oldusage
 +
temp=(application:getTextureMemoryUsage()/1024)
 +
if temp~=oldusage then
 +
oldusage=temp
 +
print("* 1 Texture usage: "..temp.." MB")
 +
end
 +
</syntaxhighlight>
 +
 +
  
  

Latest revision as of 20:25, 28 October 2024

Here are some examples from the Gideros forum and some other places :-)
All samples are ready to use. Enjoy!


LUA

TERNARY OPERATOR @hgy29

New Luau implementation here: Ternary_Operator.


Example 1: assume you have a variable called c, and you want to assign it a value only if a variable is above 10, with a ternary operator you would do this: c = a > 10 ? a : b

-- in lua you do
c = a > 10 and a or b

Example 2: assume you have a boolean called b and you want to set its default value to false

-- in lua you do
-- b = b and true or false -- that doesn't work :-/
b = b and (b ~= nil) -- that works :-)
b = b and not (b == nil) -- that also works ;-)

Example 3: assume you have a boolean called b and you want to set its default value to true

-- in lua you do
-- b = b and false or true -- that doesn't work :-/
b = b or (b == nil) -- that works :-)

STRING

A COLORED STRING @hgy29

local text = TextField.new(nil, "This is a \e[color=#f005]semi transparent red\e[color] text")
text:setPosition(32, 64)
stage:addChild(text)

A COLORED STRING 2 @mokalux

-- beautifying a tf
local tf = TextField.new(nil, "")
local mystr = "Gideros, what a beauty!"
local strsplit = mystr:split(",")
--local mystr2 = strsplit[1] -- gets the first index of the table split (=Gideros)
--local mystr3 = "\e[color=#e5e]"..mystr2.."\e[color]" -- beautifies Gideros
--mystr = mystr:gsub(mystr2, mystr3) -- replaces mystr2 by mystr3
mystr = mystr:gsub(strsplit[1], "\e[color=#e5e]"..strsplit[1].."\e[color]") -- less code!
-- final display
tf:setText(mystr)
tf:setScale(3)
tf:setPosition(128, 128)
stage:addChild(tf)

DATE & TIME @mokalux

Gets system date and time (YYYYMMDD_HHMMSS)

-- CURRENT DATE AND TIME
local myyear = os.date("*t").year
local mymonth = os.date("*t").month; if mymonth < 10 then mymonth = "0"..mymonth end
local myday = os.date("*t").day; if myday < 10 then myday = "0"..myday end
local myhour = os.date("*t").hour; if myhour < 10 then myhour = "0"..myhour end
local mymin = os.date("*t").min; if mymin < 10 then mymin = "0"..mymin end
local mysec = os.date("*t").sec; if mysec < 10 then mysec = "0"..mysec end
local mytime = myyear..mymonth..myday.."_"..myhour..mymin..mysec
print(mytime) -- 20200309_143048

FINDING A NUMBER IN A STRING @xxx

local mystring = "Doe is 26 years old."
local age = tonumber(string.match(mystring, "%d+"))
print(age) -- outputs 26

FINDING ANY INDEXED NUMBERS IN A STRING @mokalux

local mystr = "x64y250s8"
local x, y, speed -- numbers (eg. coordinates)
local index
--
index = mystr:find("s") -- "s" index represents speed
speed = tonumber(mystr:sub(index):match("%d+")) -- find sNNN
index = mystr:find("x") -- index x
x = tonumber(mystr:sub(index):match("%d+")) -- find xNNN
index = mystr:find("y") -- index y
y = tonumber(mystr:sub(index):match("%d+")) -- find yNNNN

CHECK IF STRING X STARTS OR ENDS WITH STRING Y @lua-users wiki

local function starts_with(str, start)
    return str:sub(1, #start) == start
end

local function ends_with(str, ending)
    return ending == "" or str:sub(-#ending) == ending
end

FINDING THE LAST POSITION OF CHARACTER X IN A STRING @mokalux

mystr = "This is a sample string containing many stars *. Here is another one *. And a last one *."
for i = 1, string.len(mystr) do
	if string.sub(mystr, i, i) == "*" then
		counti += 1 -- number of stars *
		countl = i -- index of the last star * in the string
	end
end
print(counti, countl)

GET PATH @xxx

function getPath(str)
	return str:match("(.*[/\\])")
end

x = "/home/user/.local/share/app/some_file"
y = "C:\\Program Files\\app\\some_file"
print(getPath(x)) -- prints: /home/user/.local/share/app/
print(getPath(y)) -- prints: C:\Program Files\app\

GET PATH AND FILENAME @mokalux

function getPathAndFileName(xpath)
	local pathlength = xpath:len()
	local tmpfolder = xpath:match("(.*[/\\])")
	local tmpfolderlength = tmpfolder:len()
	local finalfolder = tmpfolder:sub(1, tmpfolderlength-1)
	local finalfile = xpath:sub(-(pathlength-tmpfolderlength))
	return finalfolder, finalfile
end

x = "/home/user/.local/share/app/some_file.png"
local folder, file = getPathAndFileName(x)
print(folder, file) -- prints: "/home/user/.local/share/app/", "some_file.png"

FORMAT 0xRRGGBB COLOR TO STRING @Rrraptor, @keszegh

local mycolor = 0x0000ff -- your color

-- prints "0x" as regular string, then add converted hex number with 6 leading zeros
-- keszegh
local mycolorstring = string.format("0x%06x",mycolor) --> "0x0000ff"
-- Rrraptor
local mycolorstring = ("0x%06x"):format(mycolor) --> "0x0000ff"

-- converts to hex number only
-- keszegh
local mycolorstring2 = string.format("%x",mycolor) --> "ff"
-- Rrraptor
local mycolorstring2 = ("%x"):format(mycolor) --> "ff"

SPLIT A STRING EVERY N CHARACTERS @https://stackoverflow.com/questions/76194310/how-would-i-split-a-string-once-every-n-characters

local mystr = "Hello, World!"
local splitevery = 2
for c in mystr:gmatch(('.'):rep(splitevery)) do
	print(c)
end

Note: a remaining less than n characters will be silently ignored. Could be fixed by using a numeric for loop or .? instead

GFX

DRAWING A CIRCLE (PARTICLE) @hgy29

local mkr=Particles.new()
mkr:addParticles{
	{x=0,y=0,size=100,color=0xFFFFFF}, -- outter circle
	{x=0,y=0,size=70,color=0x4071B5}, -- inner circle
}
stage:addChild(mkr)
mkr:setPosition(120,120)

DRAWING A CIRCLE @xxx

xCircle = Core.class(Sprite)

function xCircle:init(r, steps, color)
   color = color or 0xff0000
   local sin, cos, d2r = math.sin, math.cos, math.pi / 180
   local c = Shape.new()
   c:setFillStyle(Shape.SOLID, color)
   c:setLineStyle(0)
   c:beginPath()
   c:moveTo(r * sin(0 * d2r), r * cos(0 * d2r))
   for i = 0, 360, 360 / steps  do
     c:lineTo(r * sin(i * d2r), r * cos(i * d2r))
   end    
   c:endPath()
   self:addChild(c)
end

--- usage
--xCircle(10, 5, 0xFF0000)

DRAWING ANOTHER CIRCLE (stars, circles, flowers) @gemboy100

-- https://forum.giderosmobile.com/discussion/comment/58662/#Comment_58662

GemBoy100Circle = Core.class(Mesh)

local floor, pi, cos, sin = math.floor, math.pi, math.cos, math.sin

function GemBoy100Circle:init(__, r1,r2, WID, HID, COLOR, ALPHA, STEPs, PULL, CUT)
	local STEPS = STEPs or floor((pi/4)*r1)
	COLOR = COLOR or 0
	PULL = PULL or 0
	local cut = CUT or 2
	if cut <= 0 then cut = 2 end
	if STEPS < 3 then STEPS = 3 end
	if PULL > r1 then
		PULL = 0
	elseif PULL > 0 then
		while STEPS%cut ~= 0 do
			STEPS+=1
		end
	end
	local c = {}
	local v = {}
	local ind = {}
	local _i = 0
	local _l = 0
	local ou = 2
	local step = pi/(STEPS/2)
	local pull = 0
	local xpull = 0
	local ypull = 0
	local widen = WID or 0
	local heden = HID or 0

	v[#v+1] = 0; v[#v+1] = 0
	c[#c+1]= COLOR; c[#c+1]= ALPHA

	for i=1, STEPS do
		_l+=1
		if _l%cut==0 then pull = PULL
		else pull = 0
		end

		local COS = cos(i*step)
		local SIN = sin(i*step)

		if COS > 0 then xpull = widen/2
		elseif COS < 0 then xpull = -widen/2 else xpull = 0 end

		if SIN > 0 then ypull = heden/2
		elseif SIN < 0 then ypull = -heden/2 else ypull = 0 end

		v[#v+1] = (r1-pull)* COS +xpull
		v[#v+1] = (r2-pull)* SIN +ypull

		v[#v+1] = (r1-pull+ou)* COS +xpull
		v[#v+1] = (r2-pull+ou)* SIN +ypull

		c[#c+1]= COLOR
		c[#c+1]= ALPHA

		c[#c+1]= 0
		c[#c+1]= 0

		ind[#ind+1] = _i+3
		ind[#ind+1] = _i+5
		ind[#ind+1] = _i+4	

		ind[#ind+1] = _i+4
		ind[#ind+1] = _i+3
		ind[#ind+1] = _i+2

		ind[#ind+1] = _i+4
		ind[#ind+1] = _i+2
		ind[#ind+1] = 1

		_i = _i+2
	end

	ind[#ind-7]=2
	ind[#ind-6]=3
	ind[#ind-5]=2
	ind[#ind-2]=2

	self:setVertexArray(v)
	self:setIndexArray(ind)
	self:setColorArray(c)
end

function GemBoy100Circle:setColor(c, a)
	a = a or 1
	self.c[1] = c
	self.c[2] = a
	for i = 3, #self.c, 4 do
		self.c[i+0] = c
		self.c[i+1] = a
	end
	self:setColorArray(self.c)
end

-- examples
--GemBoy100Circle.new(false, 32,32, 0,0, random(0xffffff),1, 32, 0, 0) -- circle
--GemBoy100Circle.new(false, 48,32, 0,0, random(0xffffff),1, 80, 0, 0) -- oval
--GemBoy100Circle.new(false, 0,0, 100,64, random(0xffffff),1, 4, 0, 0) -- rectangle
--GemBoy100Circle.new(false, 0,0, 100,64, random(0xffffff),1, 3, 0, 0) -- triangle
--GemBoy100Circle.new(false, 32,32, 16,16, random(0xffffff),1, 32, 0, 0) -- rounded cube
--GemBoy100Circle.new(false, 32,32, 48,16, random(0xffffff),1, 32, 0, 0) -- rounded rectangle
--GemBoy100Circle.new(false, 32,32, 0,0, random(0xffffff),1, 9, 16, 0) -- star
--GemBoy100Circle.new(false, 48,48, 0,0, random(0xffffff),1, 21, 32, 0) -- sun
--GemBoy100Circle.new(false, 32,32, 0,0, random(0xffffff),1, 128, 16, 8) -- sun2
--GemBoy100Circle.new(false, 32,32, 0,0, random(0xffffff),1, 12, 12, 0) -- sheriff
--GemBoy100Circle.new(false, 32,32, 0,0, random(0xffffff),1, 16, 14, 3) -- red hot
--GemBoy100Circle.new(false, 32,32, 0,0, random(0xffffff),1, 32, 8, 3) -- flower1
--GemBoy100Circle.new(false, 32,32, 0,0, random(0xffffff),1, 32, -4, 3) -- flower2

--buttons
local c1 = GemBoy100Circle.new(false, 15,15, 64,32, 0xffaa55,1, 80)
local c1bg = GemBoy100Circle.new(false, 20,20, 64,32, 0x885522,0.7, 80)
c1bg:setPosition(64*6, 64*6.5)
c1:setPosition(64*6, 64*6.5)
stage:addChild(c1bg)
stage:addChild(c1)

local c2 = GemBoy100Circle.new(false, 40,40, 32,16, 0xffaa55,1, 80)
local c2bg = GemBoy100Circle.new(false, 20,20, 80,64, 0x885522,0.7, 80)
c2bg:setPosition(64*8.5, 64*6.5)
c2:setPosition(64*8.5, 64*6.5)
stage:addChild(c2bg)
stage:addChild(c2)

local c3 = GemBoy100Circle.new(false, 32,32, 64,0, 0xffaa55,1, 80)
local c3bg = GemBoy100Circle.new(false, 36,36, 64,0, 0x885522,0.7, 80)
c3bg:setPosition(64*11, 64*6.5)
c3:setPosition(64*11, 64*6.5)
stage:addChild(c3bg)
stage:addChild(c3)

DRAWING AN ARC @hgvyas123

function drawArc(xc,yc,xradius,yradius,startAngle,endAngle,isFill)
   if yradius == nil then
      yradius = xradius
   end
   if startAngle == nil then
      startAngle = 0
   end
   if endAngle == nil then
      endAngle = 360
   end
   if isFill == nil then
      isFill = true
   end

   local shape = Shape.new()
   if isFill then
      shape:setFillStyle(Shape.SOLID, 0xff0000, 0.5)
   else
      shape:setLineStyle(3, 0x000000)
   end
   shape:beginPath()
   for i=startAngle,endAngle do
      if i == 1 then
         shape:moveTo(math.sin(math.rad(i)) * xradius, math.cos(math.rad(i)) * yradius)
      else
         shape:lineTo(math.sin(math.rad(i)) * xradius, math.cos(math.rad(i)) * yradius)
      end
   end
   if isFill then
      shape:closePath()
   end
   shape:endPath()

   shape:setPosition(xc, yc)
   return shape
end

--- usage
--function drawArc(xc,yc,xradius,yradius,startAngle,endAngle,isFill)
local myarc = drawArc(128, 128, 64, 64, 0, 5 * 45, false)
stage:addChild(myarc)

CIRCULAR PROGRESS @hgy29

CircularProgress = Core.class(Sprite)
 
function CircularProgress:init(radius, width, color, font, fontscale)
   self.radiusMin, self.radiusMax = radius - width / 2, radius + width / 2
   self.color = color
   self.font = font
   self:addChild(Path2D.new())
   local text = TextField.new(font, "-")
   text:setTextColor(color)
   text:setScale(fontscale or 1)
   self:addChild(text)
--   self:setValue(0.4)
end

function CircularProgress:setValue(val, xcolor, xalpha)
   local angs, ange = 0, math.pi * val * 1.999
   local rl, rh = self.radiusMin, self.radiusMax
   local ta = self:getChildAt(1)
   local ra = math.floor(val * 2)
--   ta:setLineColor(self.color, xalpha)
--   ta:setFillColor(self.color, xalpha)
   ta:setLineColor(xcolor, xalpha)
   ta:setFillColor(xcolor, xalpha)
   ta:setPath("MALAZ",
      math.sin(angs) * rl, -math.cos(angs) * rl,
      rl, rl, 0, ra, 1,
      math.sin(ange) * rl, -math.cos(ange) * rl,
      math.sin(ange) * rh, -math.cos(ange) * rh,
      rh, rh, 0, ra, 0,
      math.sin(angs) * rh, -math.cos(angs) * rh
      )
   ta:setLineThickness(4, 0.5)
   ta = self:getChildAt(2)
   ta:setText(("%d%%"):format(math.floor(val * 100)))
   ta:setTextColor(xcolor)
   ta:setAlpha(xalpha)
   local tax, tay, taw, tah = ta:getBounds(ta)
   ta:setAnchorPosition(tax + taw / 2, tay + tah / 2)
   return ta
end

--- usage
-- meter = CircularProgress.new(40, 5, 0x770000, nil, 2)

HEX TO RGB @rrraptor

function hex2rgb(hex)
	local r = (hex >> 16 & 0xff) / 255
	local g = (hex >> 8 & 0xff) / 255
	local b = (hex & 0xff) / 255
	return r,g,b
end

--- usage
--local r, g, b = hex2rgb(0xFF0000)

HEX TO RGB table @rrraptor

function hex2rgb(hex)
	local rgbtable = {} -- new 211115 XXX
	rgbtable.r, rgbtable.g, rgbtable.b =
		(hex >> 16 & 0xff) / 255, (hex >> 8 & 0xff) / 255, (hex & 0xff) / 255
	return rgbtable
end

--- usage
--local rgbtable = hex2rgb(0xFF0000) -- rgbtable.r, rgbtable.g, rgbtable.b

HEX TO RGB2 @https://gist.github.com/jasonbradley/4357406

function hex2rgb2(hex)
    hex = hex:gsub("#","")
    return tonumber("0x"..hex:sub(1,2)), tonumber("0x"..hex:sub(3,4)), tonumber("0x"..hex:sub(5,6))
end
--- usage
--local r, g, b = hex2rgb2(0xFF0000)

HEX TO RGB3 @keszegh

function hexToRgb(hex)
	local blue = hex%256
	local green = (hex-blue)/256 %256
	local red = (hex-blue-256*green)/256^2
	return red, green, blue
end

--- usage
--local r, g, b = hexToRgb(0x55ffcc)

RGB TO HEX @rrraptor

function rgb2hex(r, g, b)
	return (r << 16) + (g << 8) + b
end

--- usage
--local hex = rgb2hex(128, 96, 32)
--- if you want to print/display hex number
--local hexstr = string.format("0x%06x", hex)
--local hexnum = tonumber(string.format("0x%06x", hex))

RGB TO HEX2 @keszegh

function rgbToHex(r, g, b)
	return r*256*256 + g*256 + b
end

--- usage
----local hex = rgbToHex(0.5*255, 0*255, 1*255)
--local hex = rgbToHex(123, 321, 124)
--application:setBackgroundColor(hex)

PICKING A RANDOM COLOR @rrraptor

function randomColor() return "0x"..(math.random() * (1 << 24) | 0) end

--- usage
--local pixel = Pixel.new(randomColor(), 1, 32, 64)
--stage:addChild(pixel)

SAVING IMAGE TO FILE @mokalux

Saving an image or part of an image to disk.

function xsaveJPGtoDoc(xoriginalimgfile, xposx, xposy, xsizew, xsizeh, xdestimgname)
   local mytexture = Texture.new(xoriginalimgfile)
   local mybmp = Bitmap.new(mytexture)
   local rt = RenderTarget.new(mybmp:getWidth(), mybmp:getHeight())
   rt:draw(mybmp)
   rt:save("|D|tex_"..xdestimgname..".jpg", xposx, xposy, xsizew, xsizeh)
   
   return "|D|tex_"..xdestimgname..".jpg"
end

--- usage
--local xsavedjpg = xsaveJPGtoDoc("gfx/bg/dark.jpg", 0, 0, 64, 64, "dark")

PREVENT FROM DRAGGING A SHAPE OVER OTHER SHAPE @hgvyas123

--[[
Drag the shapes around with your mouse or fingers
This code is MIT licensed, see <a href="http://www.opensource.org/licenses/mit-license.php" target="_blank" rel="nofollow">http://www.opensource.org/licenses/mit-license.php</a>
(C) 2010 - 2011 Gideros Mobile
]]
 
local myShapes = {}

function Sprite:collidesWith(sprite2)
   local x,y,w,h = self:getBounds(stage)
   local x2,y2,w2,h2 = sprite2:getBounds(stage)

   return not ((y+h < y2) or (y > y2+h2) or (x > x2+w2) or (x+w < x2))
end

local function onMouseDown(self, event)
   if self:hitTestPoint(event.x, event.y) then
      self.isFocus = true
      self.x0 = event.x
      self.y0 = event.y

      event:stopPropagation()
   end
end

local function onMouseMove(self, event)
   if self.isFocus then
      self.lastX,self.lastY = self:getPosition()
      local dx = event.x - self.x0
      local dy = event.y - self.y0
      self:setX(self:getX() + dx)
      self:setY(self:getY() + dy)
      self.x0 = event.x
      self.y0 = event.y
      for i=1,#myShapes do
         if myShapes[i] ~= self then
            if self:collidesWith(myShapes[i]) then
               self:setPosition(self.lastX,self.lastY)
            end
         end
      end
      event:stopPropagation()
   end
end

local function onMouseUp(self, event)
   if self.isFocus then
      self.isFocus = false
      event:stopPropagation()
   end
end

for i=1,5 do
   local shape = Shape.new()
   shape:setLineStyle(3, 0x000000)
   shape:setFillStyle(Shape.SOLID, 0xff0000, 0.5)
   shape:beginPath()
   shape:moveTo(0, 0)
   shape:lineTo(100, 0)
   shape:lineTo(100, 50)
   shape:lineTo(0, 50)
   shape:closePath()
   shape:endPath()

   shape:setX(math.random(0, 320 - 100))
   shape:setY(math.random(0, 480 - 50))

   shape.isFocus = false

   shape:addEventListener(Event.MOUSE_DOWN, onMouseDown, shape)
   shape:addEventListener(Event.MOUSE_MOVE, onMouseMove, shape)
   shape:addEventListener(Event.MOUSE_UP, onMouseUp, shape)

   stage:addChild(shape)
   table.insert(myShapes, shape)
end

local info = TextField.new(nil, "drag the shapes around with your mouse or fingers")
info:setPosition(23, 50)
stage:addChild(info)

CHECK POINT IS IN POLYGON @atilim

function pointInPolygon(vertices, x, y)
   local intersectionCount = 0

   local x0 = vertices[#vertices].x - x
   local y0 = vertices[#vertices].y - y

   for i= 1, #vertices do
      local x1 = vertices[i].x - x
      local y1 = vertices[i].y - y

      if y0 > 0 and y1 <= 0 and x1 * y0 > y1 * x0 then
         intersectionCount = intersectionCount + 1
      end

      if y1 > 0 and y0 <= 0 and x0 * y1 > y0 * x1 then
         intersectionCount = intersectionCount + 1
      end

      x0 = x1
      y0 = y1
   end

   return (intersectionCount % 2) == 1
end

local vertices = {
   {x = 0, y = 0},
   {x = 0, y = 100},
   {x = 100, y = 100},
   {x = 0, y = 0},
}

local shape = Shape.new()
shape:setFillStyle(Shape.SOLID, 0xff0000)
shape:setLineStyle(5, 0x0000ff, 1)
shape:beginPath()
shape:moveTo(vertices[1].x,vertices[1].y)
shape:lineTo(vertices[2].x,vertices[2].y)
shape:lineTo(vertices[3].x,vertices[3].y)
shape:lineTo(vertices[4].x,vertices[4].y)
shape:endPath()

shape:setPosition(64, 64)
stage:addChild(shape)

local vertices_shape = {}
for i = 1, #vertices do
   vertices_shape[i] = {x = vertices[i].x + shape:getX(), y = vertices[i].y + shape:getY()}
end

shape:addEventListener(Event.MOUSE_HOVER, function(e)
   if pointInPolygon(vertices_shape, e.x, e.y) then
      print("hovering the shape")
   else
      print("not hovering the shape")
   end
end)

STENCIL OPERATION @hgy29

Stencil operation ex01.png

local star = Bitmap.new(Texture.new("gfx/star.png", true))
stage:addChild(star)

local bluestar_tex = RenderTarget.new(star:getWidth(), star:getHeight())
local bluestar = Bitmap.new(bluestar_tex)
stage:addChild(bluestar) bluestar:setPosition(0, 128)

-- Here is the magic!
-- Clear our render target to all alpha
bluestar_tex:clear(0x000000, 0)
--[[Gideros don't draw pixels with alpha value of 0,
they are discarded at early stage. Use the stencil to get a
footprint of visible pixels of source image]]
star:setStencilOperation {
	stencilClear = true,
	stencilWriteMask = 1,
	stencilFunc = Sprite.STENCIL_ALWAYS,
	depthPass = Sprite.STENCIL_INCR
}
bluestar_tex:draw(star)
-- Render the footprint in the color we want
local mask = Pixel.new(0x00FFFF, 1, star:getWidth(), star:getHeight())
mask:setStencilOperation {
	stencilClear = false,
	stencilMask = 1,
	stencilWriteMask = 1, 
	stencilRef = 0,
	stencilFunc = Sprite.STENCIL_NOTEQUAL,
	depthPass = Sprite.STENCIL_KEEP,
	stencilFail = Sprite.STENCIL_KEEP
}
bluestar_tex:draw(mask)
 
--[[ The result is possibly enough here, but we lost the smoothed edges of
the original texture in the process. We can restore them with a smart alpha
blending trick:]]
star:setBlendMode(Sprite.ZERO, Sprite.SRC_ALPHA)
bluestar_tex:draw(star)
 
-- Restore our initial bitmap blending mode, to show it on screen as usual
star:setBlendMode(Sprite.ALPHA)

ZOOM SPRITE TO MOUSE POSITION @hgy29, @keszegh

This one is just awesome!

application:setBackgroundColor(0x333333)

-- sprites
local canvaslayer = Sprite.new()
local pixel = Bitmap.new(Texture.new("gfx/pot/1K-triangle_buble-diffuse.jpg"))
-- order
canvaslayer:addChild(pixel)
canvaslayer:setAnchorPoint(0.5, 0.5)
stage:addChild(canvaslayer)
-- position
canvaslayer:setPosition(application:getContentWidth()/2, application:getContentHeight()/2)

-- current zoom
local currzoom = 0.5
local zoomstart = currzoom
canvaslayer:setScale(currzoom)

-- thank you very much hgy29 & keszegh :-)
local pointerstartx, pointerstarty = 0, 0
local canvasposx, canvasposy = 0, 0
local offsetx, offsety = 0, 0

stage:addEventListener(Event.MOUSE_DOWN, function(e)
	pointerstartx, pointerstarty = stage:globalToLocal(e.rx, e.ry)
	e:stopPropagation()
end)
stage:addEventListener(Event.MOUSE_MOVE, function(e)
	canvasposx, canvasposy = canvaslayer:getPosition()
	offsetx = e.rx - pointerstartx
	offsety = e.ry - pointerstarty
	canvasposx += offsetx
	canvasposy += offsety
	canvaslayer:setPosition(canvasposx, canvasposy)
	pointerstartx, pointerstarty = stage:globalToLocal(e.rx, e.ry)
	e:stopPropagation()
end)
stage:addEventListener(Event.MOUSE_WHEEL, function(e) -- e.wheel = +- 120
	zoomstart = currzoom
	pointerstartx, pointerstarty = stage:globalToLocal(e.rx, e.ry)
	canvasposx, canvasposy = canvaslayer:getPosition()
	-- local viewportZoom=math.clamp(self.zoomStart*2^((self.panStart.ry-newY)/100),_MIN_ZOOM,_MAX_ZOOM)
	currzoom = zoomstart + e.wheel/120/4
	if currzoom <= 0.2 then currzoom = 0.2 end
	if currzoom >= 2 then currzoom = 2 end
	canvaslayer:setScale(currzoom)
	local viewportx = canvasposx + (pointerstartx - canvasposx) * (1 - currzoom/zoomstart)
	local viewporty = canvasposy + (pointerstarty - canvasposy) * (1 - currzoom/zoomstart)
	canvaslayer:setPosition(viewportx, viewporty)
	e:stopPropagation()
end)

VFX

PARTICLES EFFECT 1 @hgy29

function EffectExplode(s, scale, x, y, r, speed, texture)
	local p = Particles.new()
	p:setPosition(x, y)
	p:setTexture(texture)
	p:setScale(scale)
	s:addChild(p)
	local parts = {}
	for i = 1, 50 do
		local a = math.random() * 6.3
		local dx, dy = math.sin(a), math.cos(a)
		local sr = math.random() * r
		local px, py = dx * sr, dy * sr
		local ss = (speed or 1) * (1 + math.random())
		table.insert(parts,
			{
				x = px, y = py,
				speedX = dx * ss,
				speedY = dy * ss,
				speedAngular = math.random() * 4 - 2,
				decayAlpha = 0.95 + math.random() *0.04,
				ttl = 1000,
				size = 10 + math.random() * 20
			}
		)
	end
	p:addParticles(parts)
	Core.yield(2)
	p:removeFromParent()
end

-- usage
stage:addEventListener(Event.MOUSE_DOWN,function (e)
	Core.asyncCall(EffectExplode, stage,
		1, e.x, e.y, 100, 2,
		Texture.new("gfx/smoke.png", true)
	)
end)

ANIMATION

FPS @atilim

local fps = TextField.new(nil, "")
fps:setScale(2)
fps:setPosition(16, 16)
fps:setTextColor(0xff0000)
stage:addChild(fps)

local frame = 0
local timer = os.timer()
local qFloor = math.floor
local function displayFps(event)
   frame = frame + 1
   if frame == 60 then
      local currentTimer = os.timer()
      fps:setText(qFloor(60 / (currentTimer - timer)))
      frame = 0
      timer = currentTimer
      -- optional
      --print ("memory used: "..qFloor(collectgarbage("count")/1000),"Mb")
   end
end

-- Add FPS Counter to stage
stage:addEventListener(Event.ENTER_FRAME, displayFps)

MOVING A SPRITE IN AN ARC PATH @Disciple

local bone = Sprite.new()
--local boneImg = Bitmap.new(Texture.new("gfx/enemy01.png")) -- add your gfx!
-- or use this one (Pixel)
local boneImg = Pixel.new(0x0000ff, 0.75, 32, 48)
stage:addChild(bone)
bone:addChild(boneImg)
boneImg:setAnchorPoint(0.5, 0.5)

local grav = -15 / 2
local velx = -3 / 2
local rotvel = -3 / 2
local x = 300 / 2
local y = 400 / 2
local rotation = 0

function enterFrame ()
   grav = grav + 0.3
   x = x + velx
   y = y + grav
   rotation = rotation + rotvel
   bone:setRotation(rotation)
   bone:setPosition(x, y)
end

stage:addEventListener(Event.ENTER_FRAME, enterFrame)

MOVING A SPRITE IN AN HORIZONTAL SINE WAVE @Harrison

local blimp1 = Pixel.new(0x0000ff, 0.5, 32, 64)
blimp1:setAnchorPoint(0.5, 0.5)
blimp1:setPosition(0, 128)
blimp1.speedx = 1.2
stage:addChild(blimp1)

function onEnterFrame(event)
   local posx = blimp1:getX() + blimp1.speedx
   local posy = blimp1:getY() + (1.5 * math.sin(0.075 * posx))
   if blimp1:getX() > application:getContentWidth() + blimp1:getWidth() / 2 then
      blimp1:setPosition(-blimp1:getWidth() / 2, 128)
      posx = blimp1:getX()
      posy = blimp1:getY()
   end
   blimp1:setPosition(posx, posy)
end

stage:addEventListener(Event.ENTER_FRAME, onEnterFrame)

MOVING A SPRITE IN A VERTICAL COS WAVE @Harrison

local blimp2 = Pixel.new(0x00ff00, 0.5, 64, 32)
blimp2:setAnchorPoint(0.5, 0.5)
blimp2:setPosition(128, 0)
blimp2.speedy = 1.3
stage:addChild(blimp2)

function onEnterFrame(event)
   local posy = blimp2:getY() + blimp2.speedy -- posy first
   local posx = blimp2:getX() + 2.5 * math.cos(0.1 * posy) -- posy first
   if blimp2:getY() > application:getContentHeight() + blimp2:getHeight() then
      blimp2:setPosition(128, -blimp2:getHeight())
      posx = blimp2:getX()
      posy = blimp2:getY()
   end
   blimp2:setPosition(posx, posy)
end

stage:addEventListener(Event.ENTER_FRAME, onEnterFrame)


FILES

CSV @hgy29

Load a CSV file.

function loadCSV(file)
   local function csvsplit(line)
      local t, w, l2 = {}, nil, nil
      while #line > 0 do
         w, l2 = line:match("\"([^\"]*)\"[,\r\n]?(.*)") -- Check for quoted string
         if not w then w, line = line:match("([^,]*)[,\r\n]?(.*)") -- Non quoted 
         else line = l2 end
         if not w then break end --Nothing or issue, break
         t[#t + 1] = w
      end
      return t
   end
   local headers, csv = nil, {}
   for line in io.lines(file) do
      f = csvsplit(line)
      if not headers then -- Assume first line is headers
         headers = {} for n, v in ipairs(f) do headers[v] = n end
      else
         csv[#csv + 1] = f
      end
   end
   csv.getField = function(self, row, field) return self[row][headers[field]] end
   return csv
end

--- usage
--csv=loadCSV("135-bis-bt.csv")
--print(csv:getField(4,"Value"))
--print(csv:getField(7,"Time")

JSON @xxx

Loading and saving preferences.

require "json" -- you must add JSON plugin to your app

function saveData(filepath, value)
   local contents = json.encode(value)
   local file = io.open("|D|"..filepath, "w") -- create file
   file:write(contents) -- save json string in file
   io.close(file)
end

function getData(filepath)
   local value
   local file = io.open("|D|"..filepath, "r")
   if file then
      local contents = file:read("*a") -- read contents
      value = json.decode(contents) -- decode json
      io.close(file)
   end
   return value
end

-- globals
g_configfilepath = "myplayground.txt"
g_language = application:getLanguage()
g_theme = "dark"
g_isaudio = true
g_level = 1

-- init prefs
local mydata = getData(g_configfilepath) -- try to read information from file

-- if no prefs file, create it
if not mydata then
   mydata = {g_language, g_theme, g_isaudio, g_level}
   saveData(g_configfilepath, mydata) -- create file and save datas
else
   g_language = mydata[1]
   g_theme = mydata[2]
   g_isaudio = mydata[3]
   g_level = mydata[4]
end

-- save prefs
function mySavePrefs(xlanguage, xtheme, xisaudio, xlevel)
   local mydata = {xlanguage or g_language, xtheme or g_theme, xisaudio or g_isaudio, xlevel or g_level}
   saveData(g_configfilepath, mydata) -- save new datas
   g_language = mydata[1]
   g_theme = mydata[2]
   g_isaudio = mydata[3]
   g_level = mydata[4]
end

--- usage
--mySavePrefs("en", "dark", true, 1)
--print(g_language)
--print(g_theme)

JSON V2 @mokalux

require "json" -- you must add JSON plugin to your app

function saveData(filepath, value)
	local contents = json.encode(value)
	local file = io.open(filepath, "w") -- create file
	file:write(contents) -- save json string in file
	io.close(file)
end

function getData(filepath)
	local result, value
	local file = io.open(filepath, "r")
	if file then
		local contents = file:read("*a") -- read contents
--		value = json.decode(contents) -- decode json
		result, value = pcall(json.decode, contents) -- try to decode json
		io.close(file)
	end
	return result, value
end

--[[
-- usage

-- init.lua
g_configfilepath = "|D|myconfigfile.txt" -- C:\Users\xxx\AppData\Roaming\myappname on Windows
-- ...

-- main.lua
function myLoadPrefs()
	local _result, mydata = getData(g_configfilepath) -- try to read information from file
	if not mydata then -- if no prefs file, create it
		print("* creating prefs data file")
		mydata = {}
		-- your data
		mydata.g_language = g_language
		mydata.g_bgcolor = g_bgcolor
		-- ...
		-- create file and save datas
		saveData(g_configfilepath, mydata)
		myLoadPrefs() -- reload!
	else -- data exist, use them!
		g_language = mydata.g_language -- global variable
		g_bgcolor = mydata.g_bgcolor or 0x333333 -- global variable
		-- ...
	end
end
function mySavePrefs()
	local mydata = {} -- clear the table
	-- prefs
	mydata.g_language = g_language
	mydata.g_bgcolor = g_bgcolor or 0x333333
	-- save new data
	saveData(g_configfilepath, mydata)
end

-- main.lua or elsewhere
myLoadPrefs() -- load initial prefs

-- use the data
application:setBackgroundColor(g_bgcolor)
-- ...
]]

MATH

Simple AABB Collision @Luiji Gist link

AABB collision between two rectangles.

-- check if box1 and box2 overlap
function CheckCollision(box1x, box1y, box1w, box1h, box2x, box2y, box2w, box2h)
    if box1x > box2x + box2w - 1 or -- is box1 on the right side of box2?
       box1y > box2y + box2h - 1 or -- is box1 under box2?
       box2x > box1x + box1w - 1 or -- is box2 on the right side of box1?
       box2y > box1y + box1h - 1    -- is b2 under b1?
    then
        return false                -- no collision.  Yay!
    else
        return true                 -- yes collision.  Ouch!
    end
end

AABB Collision with AnchorPoint(0.5, 0.5) @mokalux

Checks for AABB collision between two rectangles which have their anchor point set at (0.5, 0.5).

function Bullet:checkMiddleCollision(bullet, ship)
	local bulletx, bullety, bulletw, bulleth, shipx, shipy, shipw, shiph =
		bullet:getX(), bullet:getY(), bullet:getWidth(), bullet:getHeight(),
		ship:getX(), ship:getY(), ship:getWidth(), ship:getHeight()
	if
		(bulletx + bulletw/2 > shipx and
		bulletx < shipx + shipw/2 and
		bullety + bulleth/2 > shipy and
		bullety < shipy + shiph/2)
		or
		(bulletx - bulletw/2 < shipx and
		bulletx > shipx - shipw/2 and
		bullety - bulleth/2 < shipy and
		bullety > shipy - shiph/2)
	then
		if bullet.isplayer and not ship.isplayer or -- player bullet vs enemy ship
			not bullet.isplayer and ship.isplayer then -- enemy bullet vs player ship
				return true, bullet, ship -- collision
		end
	else
			return false -- no collision
	end
end

-- sample usage in ENTER_FRAME
function Bullet:tick(delay)
	local x,y=self:getPosition()
	x+=self.dx*self.speed
	y-=self.dy*self.speed
	self:setPosition(x,y)
	for k, _ in pairs(BULLETS) do
		for k1, _ in pairs(ACTORS) do
			local c, b, s = self:checkMiddleCollision(k, k1)
			if c then
				b:destroy()
				s:hit(self.damage)
			end
		end
	end
end

CLAMP @rrraptor

Limits the value to the specified range.

-- v - value to limit
-- min, max - range
function clamp(v, min, max)
	return (v <> min) >< max
end

--- usage
--clamp(20, 0, 100) --> 20
--clamp(-20, 0, 100) --> 0
--clamp(120, 0, 100) --> 100

LERP @xxx

Lerps between 2 values.

local function lerp(a, b, t)
	return a + (b-a) * t
end

MAP @rrraptor

Transforms one range to another.

-- v - value to map
-- minSrc - lower bound of the value's current range
-- maxSrc - upper bound of the value's current range
-- minDst - lower bound of the value's target range
-- maxDst - upper bound of the value's target range
-- clampValue - constrain the value to the newly mapped range
function map(v, minSrc, maxSrc, minDst, maxDst, clampValue)
	local newV = (v - minSrc) / (maxSrc - minSrc) * (maxDst - minDst) + minDst
	return not clampValue and newV or clamp(newV, minDst >< maxDst, minDst <> maxDst)
end

--- usage
--map(0.5, 0, 1, 0, 200) -> 100
--map(10, 0, 10, 0, 200) -> 200
--map(0, -1, 1, 0, 1) -> 0.5

ROUND @http://lua-users.org/wiki/FormattingNumbers

Rounds a number to the nearest decimal.

--[[
	round(val, n) - Rounds a number to the nearest decimal places.
	val - The value to round.
	n - Number of decimal places to round to.
	-- usage
	--print( round( 1.5678 ) ) -- prints: 1.57
]]
function xround(val, n)
	if (n) then
		return math.floor( (val * 10^n)+0.5 ) / (10^n)
	else
		return math.floor(val+0.5)
	end
end

ROUND2 @https://stackoverflow.com/a/58411671/870125

Rounds a number to the nearest decimal.

local function round(num)
	return num + (2^52 + 2^51) - (2^52 + 2^51)
end

ROUND3 @http://lua-users.org/wiki/SimpleRound

Rounds a number to the nearest decimal. Seems to be fast.

function round(n, mult)
	mult = mult or 1
	return math.floor((n+mult/2)/mult) * mult
end

Move Towards @https://devforum.roblox.com/t/new-lua-functions-mathsign-and-mathclamp/37309/14

local function moveTowards(current, target, maxDelta)
	if current < target then return math.min(target, current + maxDelta)
	elseif current > target then return math.max(target, current - maxDelta)
	else return target
	end
end

Shoot Towards @mokalux

function shoot()
	local projectiletex = "gfx/projectiles/bullet1.png"
	local projectilespeed = 8*1
	-- a turret targeting a player
	local angle = math.atan2(player.y - turret.y, player.x - turret.x)
	-- projectile direction
	local vx, vy = projectilespeed * math.cos(angle), projectilespeed * math.sin(angle)
	-- ...
	-- projectile implementation example
	local g = Projectile.new(
		texture=projectiletex,
		x=turret.x + 8*4, y=turret.y + 0, -- your projectile start location
		speedx=vx, speedy=vy,
	})
	g = nil -- cleanup
	-- ...
end

Calculate Angle between 2 Points @GregBUG

local qAtan2 = math.atan2

local function calcAngle(xA, yA, xB, yB)
	return qAtan2((yB - yA), (xB - xA))
end

Calculate Distance between 2 Points @GregBUG

local function distance(xA, yA, xB, yB)
	local dx = xB - xA
	local dy = yB - yA
--	return (dx*dx + dy*dy)^0.5 -- much faster than math.sqrt
	return (dx^2 + dy^2)^0.5 -- this is an even faster one, and the fastest you can get!
end

TABLE

SELF VALUE TABLE TRICK @mokalux

local myvalue == "myvalue"
self[myvalue] -- this is equivalent to self.myvalue

HIGHEST / LOWEST NUMBER IN A TABLE @https://stackoverflow.com/questions/5178087/how-do-i-get-the-highest-integer-in-a-table-in-lua

Gets the highest or the lowest number in a table.

print(math.max(unpack{4, 5, 62, -5, 12, 9})) -- 62
print(math.min(unpack{4, 5, 62, -5, 12, 9})) -- -5

MERGING 2 TABLES @xxx

method 1:

z = {}
n = 0
for _,v in ipairs(x) do n=n+1; z[n]=v end
for _,v in ipairs(y) do n=n+1; z[n]=v end

method 2:

levelsetup = {color=0x5E3C3A, isshape=true, restitution=0, friction=1}
tablebase = { x = 64, y = 200, w = 400, h = 250, rotation = 0}
for k, v in pairs(levelsetup) do
	tablebase[k] = v
end

NICE TABLE PRINT FORMATTING @xxx

function print_r(t, name, indent)
	local tableList = {}
	local function table_r(t, name, indent, full)
		local id = not full and name or type(name) ~= "number" and tostring(name) or "["..name.."]"
		local tag = indent .. id .. " = "
		local out = {} -- result
		if type(t) == "table" then
			if tableList[t] ~= nil then
				table.insert(out, tag .. "{} -- " .. tableList[t] .. " (self reference)")
			else
				tableList[t] = full and (full .. "." .. id) or id
				if next(t) then -- Table not empty
					table.insert(out, tag .. "{")
					for key,value in pairs(t) do
						table.insert(out, table_r(value, key, indent .. "|  ", tableList[t]))
					end
					table.insert(out, indent .. "}")
				else
					table.insert(out, tag .. "{}")
				end
			end
		else
			local val = type(t) ~= "number" and type(t) ~= "boolean" and "\""..tostring(t).."\"" or tostring(t)
			table.insert(out, tag .. val)
		end

		return table.concat(out, "\n")
	end
	print(table_r(t, name or "Value", indent or ""))
end

-- usage
a = {x=1, y=2}
print("TEST: ", a["x"])
print_r(a, "table...", " ****")

-- result
--TEST: 	1
-- ****table... = {
-- ****|  y = 2
-- ****|  x = 1
-- ****}

CHECK IF TABLE HAS DUPLICATES @mokalux

This code was written as a local function inside another function. You can adapt the below code and pass your table as parameter to make the function more global.

local function isDouble(val)
	local double = false
	for k, v in pairs(yourtable) do if v == val then double = true end end -- make sure yourtable is accesible here!
	return double
end

-- real life example: REMAP KEYS
function Options:onKeyDown(ev)
	local keycode = ev.keyCode
	local key = utf8.char(keycode)
	local keys = { left=g_keyleft, right=g_keyright, up=g_keyup, down=g_keydown, action1=g_keyaction1, }
	local function isDouble(val)
		local double = false
		for k, v in pairs(keys) do if v == val then double = true end end
		return double
	end
	if not isDouble(keycode) then self.btns[self.selector]:changeText(key) end -- check if a key was already assigned
	if self.selector == 1 then g_keyleft = keycode
	elseif self.selector == 2 then g_keyright = keycode
	elseif self.selector == 3 then g_keyup = keycode
	elseif self.selector == 4 then g_keydown = keycode
	elseif self.selector == 5 then g_keyaction1 = keycode
	end
	mySavePrefs(g_configfilepath)
	self:remapKey(false)
end

DISPLAY LIST AS GRID ALGO @sslivka

-- display list as grid algo @sslivka
local random = math.random
local shapes = {}
shapes[#shapes+1] = Pixel.new(random(0xffffff),1, 32, 32)
shapes[#shapes+1] = Pixel.new(random(0xffffff),1, 32, 32)
shapes[#shapes+1] = Pixel.new(random(0xffffff),1, 32, 32)
shapes[#shapes+1] = Pixel.new(random(0xffffff),1, 32, 32)
shapes[#shapes+1] = Pixel.new(random(0xffffff),1, 32, 32)
shapes[#shapes+1] = Pixel.new(random(0xffffff),1, 32, 32)

-- grid settings
local cellw = 32
local cellh = 32
local cellvspacing = 32*0.8
local cellhspacing = 32*0.4
local colsplit = 4 -- you choose
-- calculate number of rows
local rows, frac = math.modf(#shapes/colsplit) -- integer and fractional values
if frac > 0 then rows += 1 end -- if there is a fraction increase number of rows by 1
-- grid starting position
local startx, starty = 32*1, 32*1
local x, y = startx, starty
-- grid loop
for i = 1, rows do
	y += (i-1)*cellhspacing
	for j = 1, colsplit do
		if (i-1)*colsplit + j > #shapes then return end
		x += cellvspacing - (j*cellvspacing)
		local ci = (i-1)*colsplit + j -- column index
		shapes[ci]:setPosition(x, y)
		stage:addChild(shapes[ci])
		x += cellw + (j*cellvspacing) + cellvspacing
	end
	x = startx
	y += cellh - (i-1)*cellhspacing
end

DEBUGGING

TEXTURE MEMORY USAGE @SinisterSoft

local temp, oldusage
temp=(application:getTextureMemoryUsage()/1024)
if temp~=oldusage then
	oldusage=temp
	print("* 1 Texture usage: "..temp.." MB")
end



More to come God's willing...