Merge pull request #76 from duncathan/sharecode

Adds sharecodes
This commit is contained in:
duncathan salt 2020-02-28 04:54:34 -06:00 committed by GitHub
commit 4e1dbd19cf
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
6 changed files with 400 additions and 31 deletions

260
src/lib/bit.lua Normal file
View file

@ -0,0 +1,260 @@
--[[---------------
LuaBit v0.4
-------------------
a bitwise operation lib for lua.
http://luaforge.net/projects/bit/
How to use:
-------------------
bit.bnot(n) -- bitwise not (~n)
bit.band(m, n) -- bitwise and (m & n)
bit.bor(m, n) -- bitwise or (m | n)
bit.bxor(m, n) -- bitwise xor (m ^ n)
bit.brshift(n, bits) -- right shift (n >> bits)
bit.blshift(n, bits) -- left shift (n << bits)
bit.blogic_rshift(n, bits) -- logic right shift(zero fill >>>)
Please note that bit.brshift and bit.blshift only support number within
32 bits.
2 utility functions are provided too:
bit.tobits(n) -- convert n into a bit table(which is a 1/0 sequence)
-- high bits first
bit.tonumb(bit_tbl) -- convert a bit table into a number
-------------------
Under the MIT license.
copyright(c) 2006~2007 hanzhao (abrash_han@hotmail.com)
--]]---------------
do
------------------------
-- bit lib implementions
local function check_int(n)
-- checking not float
if(n - math.floor(n) > 0) then
error("trying to use bitwise operation on non-integer!")
end
end
local function to_bits(n)
check_int(n)
if(n < 0) then
-- negative
return to_bits(bit.bnot(math.abs(n)) + 1)
end
-- to bits table
local tbl = {}
local cnt = 1
while (n > 0) do
local last = math.mod(n,2)
if(last == 1) then
tbl[cnt] = 1
else
tbl[cnt] = 0
end
n = (n-last)/2
cnt = cnt + 1
end
return tbl
end
local function tbl_to_number(tbl)
local n = table.getn(tbl)
local rslt = 0
local power = 1
for i = 1, n do
rslt = rslt + tbl[i]*power
power = power*2
end
return rslt
end
local function expand(tbl_m, tbl_n)
local big = {}
local small = {}
if(table.getn(tbl_m) > table.getn(tbl_n)) then
big = tbl_m
small = tbl_n
else
big = tbl_n
small = tbl_m
end
-- expand small
for i = table.getn(small) + 1, table.getn(big) do
small[i] = 0
end
end
local function bit_or(m, n)
local tbl_m = to_bits(m)
local tbl_n = to_bits(n)
expand(tbl_m, tbl_n)
local tbl = {}
local rslt = math.max(table.getn(tbl_m), table.getn(tbl_n))
for i = 1, rslt do
if(tbl_m[i]== 0 and tbl_n[i] == 0) then
tbl[i] = 0
else
tbl[i] = 1
end
end
return tbl_to_number(tbl)
end
local function bit_and(m, n)
local tbl_m = to_bits(m)
local tbl_n = to_bits(n)
expand(tbl_m, tbl_n)
local tbl = {}
local rslt = math.max(table.getn(tbl_m), table.getn(tbl_n))
for i = 1, rslt do
if(tbl_m[i]== 0 or tbl_n[i] == 0) then
tbl[i] = 0
else
tbl[i] = 1
end
end
return tbl_to_number(tbl)
end
local function bit_not(n)
local tbl = to_bits(n)
local size = math.max(table.getn(tbl), 32)
for i = 1, size do
if(tbl[i] == 1) then
tbl[i] = 0
else
tbl[i] = 1
end
end
return tbl_to_number(tbl)
end
local function bit_xor(m, n)
local tbl_m = to_bits(m)
local tbl_n = to_bits(n)
expand(tbl_m, tbl_n)
local tbl = {}
local rslt = math.max(table.getn(tbl_m), table.getn(tbl_n))
for i = 1, rslt do
if(tbl_m[i] ~= tbl_n[i]) then
tbl[i] = 1
else
tbl[i] = 0
end
end
--table.foreach(tbl, print)
return tbl_to_number(tbl)
end
local function bit_rshift(n, bits)
check_int(n)
local high_bit = 0
if(n < 0) then
-- negative
n = bit_not(math.abs(n)) + 1
high_bit = 2147483648 -- 0x80000000
end
for i=1, bits do
n = n/2
n = bit_or(math.floor(n), high_bit)
end
return math.floor(n)
end
-- logic rightshift assures zero filling shift
local function bit_logic_rshift(n, bits)
check_int(n)
if(n < 0) then
-- negative
n = bit_not(math.abs(n)) + 1
end
for i=1, bits do
n = n/2
end
return math.floor(n)
end
local function bit_lshift(n, bits)
check_int(n)
if(n < 0) then
-- negative
n = bit_not(math.abs(n)) + 1
end
for i=1, bits do
n = n*2
end
return bit_and(n, 4294967295) -- 0xFFFFFFFF
end
local function bit_xor2(m, n)
local rhs = bit_or(bit_not(m), bit_not(n))
local lhs = bit_or(m, n)
local rslt = bit_and(lhs, rhs)
return rslt
end
--------------------
-- bit lib interface
bit = {
-- bit operations
bnot = bit_not,
band = bit_and,
bor = bit_or,
bxor = bit_xor,
brshift = bit_rshift,
blshift = bit_lshift,
bxor2 = bit_xor2,
blogic_rshift = bit_logic_rshift,
-- utility func
tobits = to_bits,
tonumb = tbl_to_number,
}
end
--[[
for i = 1, 100 do
for j = 1, 100 do
if(bit.bxor(i, j) ~= bit.bxor2(i, j)) then
error("bit.xor failed.")
end
end
end
--]]

View file

@ -12,6 +12,7 @@ ld = love.data
U = require 'util'
require 'log'
require 'lib.bit'
local random = require 'randomizer'
local settings = require 'settings'

View file

@ -35,6 +35,7 @@ function C:new()
self.customseed = nil
self.puppy = false
self.obj = ""
self.sharecode = ""
end
function C:setPath(path)
@ -53,7 +54,11 @@ function C:randomize()
return "Could not find \"data\" subfolder.\n\nMaybe try dropping your Cave Story \"data\" folder in directly?"
end
self:_logSettings()
local seed = self:_seedRngesus()
self:_updateSharecode(seed)
local tscFiles = self:_createTscFiles(dirStage)
-- self:_writePlaintext(tscFiles)
self:_shuffleItems(tscFiles)
@ -64,7 +69,7 @@ function C:randomize()
self:_updateSettings()
return self:_getStatusMessage(seed)
return self:_getStatusMessage(seed, self.sharecode)
end
function C:_mountDirectory(path)
@ -146,8 +151,6 @@ function C:_shuffleItems(tscFiles)
-- place the bomb on MALCO for bad end
if self.obj == "objBadEnd" then
self.worldGraph:getMALCO()[1]:setItem(self.itemDeck:getByKey("bomb"))
else
logWarning(self.obj)
end
@ -257,17 +260,57 @@ function C:_unmountDirectory(path)
assert(lf.unmount(path))
end
function C:_logSettings()
local obj = "Best Ending"
if self.obj == "objBadEnd" then
obj = "Bad Ending"
elseif self.obj == "objNormalEnd" then
obj = "Normal Ending"
elseif self.obj == "objAllBosses" then
obj = "All Bosses"
end
local puppy = self.puppy and ", Puppysanity" or ""
logNotice(("Game settings: %s"):format(obj .. puppy))
end
function C:_updateSettings()
Settings.settings.puppy = self.puppy
Settings.settings.obj = self.obj
Settings:update()
end
function C:_getStatusMessage(seed)
function C:_updateSharecode(seed)
local settings = 0 -- 0b00000000
-- 0bXXXXXPOO
-- P: single bit used for puppysanity
-- O: two bits used for objective
-- X: unused
if self.obj == "objBadEnd" then
settings = bit.bor(settings, 1)
elseif self.obj == "objNormalEnd" then
settings = bit.bor(settings, 2)
elseif self.obj == "objAllBosses" then
settings = bit.bor(settings, 3)
end
if self.puppy then settings = bit.bor(settings, 4) end
if #seed < 20 then
seed = seed .. (" "):rep(20-#seed)
end
local packed = love.data.pack("data", "sB", seed, settings)
self.sharecode = love.data.encode("string", "base64", packed)
logNotice(("Sharecode: %s"):format(self.sharecode))
end
function C:_getStatusMessage(seed, sharecode)
local warnings, errors = countLogWarningsAndErrors()
local line1
if warnings == 0 and errors == 0 then
line1 = ("Randomized data successfully created!\nSeed: %s"):format(seed)
line1 = ("Randomized data successfully created!\nSeed: %s\nSharecode: %s"):format(seed, sharecode)
elseif warnings ~= 0 and errors == 0 then
line1 = ("Randomized data was created with %d warning(s)."):format(warnings)
else
@ -275,7 +318,7 @@ function C:_getStatusMessage(seed)
end
local line2 = "Next overwrite the files in your copy of Cave Story with the versions in the newly created \"data\" folder. Don't forget to save a backup of the originals!"
local line3 = "Then play and have a fun!"
local status = ("%s\n%s\n\n%s"):format(line1, line2, line3)
local status = ("%s\n%s\n%s"):format(line1, line2, line3)
return status
end

View file

@ -12,19 +12,27 @@ layout:setTheme(require 'lib.luigi.theme.dark')
settings:setTheme(require 'lib.luigi.theme.dark')
function C:setup()
settings.puppy.value = Settings.settings.puppy
local obj = Settings.settings.obj
if obj == "objBadEnd" then
self:loadSettings(Settings.settings.puppy, Settings.settings.obj)
background = lg.newImage('assets/background.png')
self:draw()
layout:show()
end
function C:loadSettings(puppy, obj, seed)
settings.puppy.value = puppy
if obj == "objBadEnd" or obj == 1 then
settings.bad.value = true
settings.norm.value = false
settings.boss.value = false
settings.best.value = false
elseif obj == "objNormalEnd" then
elseif obj == "objNormalEnd" or obj == 2 then
settings.bad.value = false
settings.norm.value = true
settings.boss.value = false
settings.best.value = false
elseif obj == "objAllBosses" then
elseif obj == "objAllBosses" or obj == 3 then
settings.bad.value = false
settings.norm.value = false
settings.boss.value = true
@ -36,9 +44,11 @@ function C:setup()
settings.best.value = true
end
background = lg.newImage('assets/background.png')
self:draw()
layout:show()
if seed ~= nil then
settings.customseed.value = seed or ""
settings.seedselect.value = true
settings.seedrandom.value = false
end
end
layout.version.text = 'Cave Story Randomizer [Open Mode] v' .. VERSION
@ -48,9 +58,11 @@ layout.twitter.text = '(@shruuu and @duncathan_salt)'
layout.footer.text = 'Original randomizer:\r\nshru.itch.io/cave-story-randomizer'
layout.go:onPress(function()
Randomizer:new()
if Randomizer:ready() then
if settings.seedselect.value and settings.customseed.value ~= "" then
Randomizer.customseed = settings.customseed.value
Randomizer.customseed = settings.customseed.value:gsub("^%s*(.-)%s*$", "%1") -- trim any leading/trailing whitespace
end
if settings.bad.value then
@ -65,7 +77,8 @@ layout.go:onPress(function()
Randomizer.puppy = settings.puppy.value
C:setStatus(Randomizer:randomize())
Randomizer:new()
layout.sharecode.text = "Copy Sharecode"
else
C:setStatus("No Cave Story folder found!\r\nDrag and drop your Cave Story folder here.")
end
@ -79,6 +92,43 @@ end)
settings.closeButton:onPress(function()
settings:hide()
layout:show()
settings.sharecode.value = ""
end)
layout.sharecode:onPress(function()
if Randomizer.sharecode ~= "" then
love.system.setClipboardText(Randomizer.sharecode)
layout.sharecode.text = "Copied!"
end
end)
settings.customseed:onChange(function()
if #settings.customseed.value > 20 then
settings.customseed.value = settings.customseed.value:sub(1, 20)
end
settings.seedcount.text = ("%s/20"):format(#settings.customseed.value)
end)
settings.importshare:onPress(function()
local success, seed, sharesettings = pcall(function()
local packed = love.data.decode("data", "base64", settings.sharecode.value)
local seed, settings = love.data.unpack("sB", packed)
return seed, settings
end)
if success then
settings.importshare.text = "Sharecode Imported"
local pup = bit.band(sharesettings, 4) ~= 0
local obj = bit.band(sharesettings, 3)
seed = seed:gsub("^%s*(.-)%s*$", "%1") -- trim any leading or trailing whitespace
Screen:loadSettings(pup, obj, seed)
else
settings.importshare.text = "Invalid Sharecode!"
end
end)
settings.sharecode:onChange(function()
settings.importshare.text = "Import Sharecode"
end)
function C:draw()

View file

@ -37,7 +37,15 @@ return { id = 'window',
type = 'button',
style = 'button',
id = 'go',
text = 'Randomize',
text = 'Randomize!',
width = 100,
height = 32,
},
{
type = 'button',
style = 'button',
id = 'sharecode',
text = "Copy Sharecode",
width = 100,
height = 32,
},

View file

@ -1,23 +1,30 @@
return { style = 'dialog',
{ style = 'dialogHead', text = 'Settings' },
{ style = 'dialogBody', padding = 24, flow = 'x',
{
{ type = 'label', text = 'Seed' },
{ style = 'dialogBody', padding = 24,
{ flow = 'x', minheight = 260,
{
{ type = 'radio', group = 'seed', text = 'Use random seed', value = true },
{ flow = 'y', { type = 'radio', group = 'seed', text = 'Use custom seed', id = 'seedselect'}, { type = 'text', id = 'customseed', width = 150 }, {height = false} }
{ type = 'label', text = 'Seed', minheight = 32 },
{
{ type = 'radio', group = 'seed', text = 'Use random seed', value = true, id = 'seedrandom', minheight = 27 },
{ flow = 'y', { type = 'radio', group = 'seed', text = 'Use custom seed', id = 'seedselect', minheight = 27 }, {{ type = 'text', id = 'customseed', width = 200, minheight = 32}, flow = 'x', { type = 'label', id = 'seedcount' }} }
},
{ type = 'label', text = 'Randomization Options', minheight = 32 },
{ type = 'check', value = false, id = 'puppy', text = "Puppysanity", minheight = 27 },
{ height = 64 },
},
{
{ type = 'label', text = 'Objective', minheight = 32 },
{ type = 'radio', group = 'objective', text = 'Bad ending', id = 'bad', minheight = 27 },
{ type = 'radio', group = 'objective', text = 'Normal ending', id = 'norm', minheight = 27 },
{ type = 'radio', group = 'objective', text = 'Best ending', id = 'best', value = true, minheight = 27 },
{ type = 'radio', group = 'objective', text = 'All bosses', id = 'boss', minheight = 27 },
},
{ type = 'label', text = 'Randomization Options' },
{ type = 'check', value = false, id = 'puppy', text = "Puppysanity"},
{ height = false },
},
{
{ type = 'label', text = 'Objective' },
{ type = 'radio', group = 'objective', text = 'Bad ending', id = 'bad' },
{ type = 'radio', group = 'objective', text = 'Normal ending', id = 'norm'},
{ type = 'radio', group = 'objective', text = 'Best ending', id = 'best', value = true },
{ type = 'radio', group = 'objective', text = 'All bosses', id = 'boss' },
},
flow = 'x',
{ type = 'text', id = 'sharecode', width = 350 },
{ type = 'button', style = 'dialogButton', text = "Import Sharecode", id = 'importshare', width = 180 }
}
},
{ style = 'dialogFoot',
{},