Skip to content

Commit

Permalink
Merge remote-tracking branch 'origin/main' into open-cloud-luau-execu…
Browse files Browse the repository at this point in the history
…tion
  • Loading branch information
vocksel committed Nov 3, 2024
2 parents 8afcae4 + 5daea9c commit e8d4eb2
Show file tree
Hide file tree
Showing 24 changed files with 534 additions and 180 deletions.
2 changes: 1 addition & 1 deletion LICENSE
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
Copyright 2021 David Minnerly
Copyright 2021 flipbook-labs

Permission is hereby granted, free of charge, to any person obtaining a copy of
this software and associated documentation files (the "Software"), to deal in
Expand Down
5 changes: 4 additions & 1 deletion build.project.json
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,9 @@
"Example": {
"$path": "example"
},
"wally.toml": {
"$path": "wally.toml"
},
"$path": "src"
}
}
}
5 changes: 4 additions & 1 deletion default.project.json
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,9 @@
"Packages": {
"$path": "Packages"
},
"wally.toml": {
"$path": "wally.toml"
},
"$path": "build"
}
}
}
2 changes: 1 addition & 1 deletion foreman.toml
Original file line number Diff line number Diff line change
Expand Up @@ -7,5 +7,5 @@ selene = { source = "Kampfkarren/selene", version = "0.27.1" }
stylua = { source = "JohnnyMorganz/StyLua", version = "0.20.0" }
tarmac = { source = "Roblox/tarmac", version = "0.7.0" }
wally = { source = "UpliftGames/wally", version = "0.3.2" }
luau-lsp = { source = "JohnnyMorganz/luau-lsp", version = "1.32.3" }
luau-lsp = { source = "JohnnyMorganz/luau-lsp", version = "1.34.0" }
wally-package-types = { source = "JohnnyMorganz/wally-package-types", version = "1.3.2" }
Binary file added img/GitHubMark.png
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
147 changes: 147 additions & 0 deletions src/About/AboutView.luau
Original file line number Diff line number Diff line change
@@ -0,0 +1,147 @@
local React = require("@pkg/React")

local RobloxProfile = require("@root/About/RobloxProfile")
local Sprite = require("@root/Common/Sprite")
local assets = require("@root/assets")
local nextLayoutOrder = require("@root/Common/nextLayoutOrder")
local useTheme = require("@root/Common/useTheme")
local wally = require(script.Parent.Parent["wally.toml"])

local useMemo = React.useMemo

local AUTHOR_USER_IDS = {
1343930,
731053179,
}

local function AboutView()
local theme = useTheme()
local currentYear = useMemo(function()
return (DateTime.now():ToUniversalTime() :: any).Year
end, {})

local authors: { [string]: React.Node } = {}
for _, userId in AUTHOR_USER_IDS do
authors[`Author{userId}`] = React.createElement(RobloxProfile, {
userId = userId,
LayoutOrder = nextLayoutOrder(),
})
end

return React.createElement("Frame", {
Size = UDim2.fromScale(1, 1),
BackgroundColor3 = theme.background,
BorderSizePixel = 0,
}, {
Layout = React.createElement("UIListLayout", {
SortOrder = Enum.SortOrder.LayoutOrder,
HorizontalAlignment = Enum.HorizontalAlignment.Center,
Padding = theme.paddingLarge,
}),

Padding = React.createElement("UIPadding", {
PaddingTop = theme.paddingLarge,
PaddingRight = theme.paddingLarge,
PaddingBottom = theme.paddingLarge,
PaddingLeft = theme.paddingLarge,
}),

Logo = React.createElement(Sprite, {
layoutOrder = nextLayoutOrder(),
image = assets.IconLight,
size = UDim2.fromOffset(42, 42),
}),

Title = React.createElement("TextLabel", {
AutomaticSize = Enum.AutomaticSize.XY,
BackgroundTransparency = 1,
Font = theme.font,
LayoutOrder = nextLayoutOrder(),
Size = UDim2.fromScale(0, 0),
Text = `flipbook v{wally.package.version}`,
TextColor3 = theme.text,
TextSize = theme.headerTextSize,
TextWrapped = true,
}),

GitHub = React.createElement("Frame", {
AutomaticSize = Enum.AutomaticSize.XY,
BackgroundTransparency = 1,
LayoutOrder = nextLayoutOrder(),
}, {
Layout = React.createElement("UIListLayout", {
SortOrder = Enum.SortOrder.LayoutOrder,
FillDirection = Enum.FillDirection.Horizontal,
Padding = theme.paddingSmall,
}),

Icon = React.createElement(Sprite, {
layoutOrder = nextLayoutOrder(),
color = theme.github,
image = assets.GitHubMark,
size = UDim2.fromOffset(theme.textSize, theme.textSize),
}),

Label = React.createElement("TextLabel", {
AutomaticSize = Enum.AutomaticSize.XY,
BackgroundTransparency = 1,
Font = theme.font,
LayoutOrder = nextLayoutOrder(),
Size = UDim2.fromScale(0, 0),
Text = "flipbook-labs/flipbook",
TextColor3 = theme.text,
TextSize = theme.textSize,
}),
}),

Authors = React.createElement("Frame", {
AutomaticSize = Enum.AutomaticSize.XY,
BackgroundTransparency = 1,
LayoutOrder = nextLayoutOrder(),
}, {
Layout = React.createElement("UIListLayout", {
SortOrder = Enum.SortOrder.LayoutOrder,
HorizontalAlignment = Enum.HorizontalAlignment.Center,
Padding = theme.padding,
}),

Title = React.createElement("TextLabel", {
AutomaticSize = Enum.AutomaticSize.XY,
BackgroundTransparency = 1,
Font = theme.font,
LayoutOrder = nextLayoutOrder(),
Size = UDim2.fromScale(0, 0),
Text = `Created by:`,
TextColor3 = theme.text,
TextSize = theme.textSize,
TextWrapped = true,
}),

AuthorList = React.createElement("Frame", {
AutomaticSize = Enum.AutomaticSize.XY,
BackgroundTransparency = 1,
LayoutOrder = nextLayoutOrder(),
}, {
Layout = React.createElement("UIListLayout", {
SortOrder = Enum.SortOrder.LayoutOrder,
FillDirection = Enum.FillDirection.Horizontal,
Padding = theme.padding,
}),
}, authors),
}),

Copy = React.createElement("TextLabel", {
AutomaticSize = Enum.AutomaticSize.XY,
BackgroundTransparency = 1,
Font = theme.font,
LayoutOrder = nextLayoutOrder(),
Size = UDim2.fromScale(0, 0),
Text = `Copyright © 2021—{currentYear} flipbook-labs`,
TextColor3 = theme.text,
TextSize = theme.textSize,
TextWrapped = true,
}),
})
end

return AboutView
15 changes: 15 additions & 0 deletions src/About/AboutView.story.luau
Original file line number Diff line number Diff line change
@@ -0,0 +1,15 @@
local React = require("@pkg/React")

local AboutView = require("./AboutView")
local ContextProviders = require("@root/Common/ContextProviders")
local MockPlugin = require("@root/Testing/MockPlugin")

return {
story = function()
return React.createElement(ContextProviders, {
plugin = MockPlugin.new(),
}, {
AboutView = React.createElement(AboutView),
})
end,
}
79 changes: 79 additions & 0 deletions src/About/RobloxProfile.luau
Original file line number Diff line number Diff line change
@@ -0,0 +1,79 @@
local Players = game:GetService("Players")

local React = require("@pkg/React")
local ReactSpring = require("@pkg/ReactSpring")

local nextLayoutOrder = require("@root/Common/nextLayoutOrder")
local useTheme = require("@root/Common/useTheme")

local useState = React.useState
local useEffect = React.useEffect

local AVATAR_SIZE = Enum.ThumbnailSize.Size48x48

export type Props = {
userId: number,
LayoutOrder: number?,
}

local function RobloxProfile(props: Props)
local theme = useTheme()
local avatar, setAvatar = useState(nil :: string?)
local username, setUsername = useState(nil :: string?)

local isLoading = avatar == nil and username == nil

local styles = ReactSpring.useSpring({
alpha = if isLoading then 1 else 0,
})

useEffect(function()
task.spawn(function()
setAvatar(Players:GetUserThumbnailAsync(props.userId, Enum.ThumbnailType.HeadShot, AVATAR_SIZE))
end)

task.spawn(function()
setUsername(Players:GetNameFromUserIdAsync(props.userId))
end)
end, { props.userId })

return React.createElement("Frame", {
LayoutOrder = props.LayoutOrder,
AutomaticSize = Enum.AutomaticSize.XY,
BackgroundTransparency = 1,
}, {
Layout = React.createElement("UIListLayout", {
SortOrder = Enum.SortOrder.LayoutOrder,
FillDirection = Enum.FillDirection.Horizontal,
VerticalAlignment = Enum.VerticalAlignment.Center,
Padding = theme.paddingSmall,
}),

Avatar = React.createElement("ImageLabel", {
LayoutOrder = nextLayoutOrder(),
Size = UDim2.fromOffset(48, 48),
BackgroundColor3 = theme.sidebar,
BackgroundTransparency = 0.6,
ImageTransparency = styles.alpha,
BorderSizePixel = 0,
Image = avatar,
}, {
Corner = React.createElement("UICorner", {
CornerRadius = UDim.new(0.5, 0),
}),
}),

Username = React.createElement("TextLabel", {
LayoutOrder = nextLayoutOrder(),
Text = `@{username}`,
Font = theme.font,
TextColor3 = theme.text,
TextSize = theme.textSize,
BackgroundTransparency = 1,
TextTransparency = styles.alpha,
AutomaticSize = Enum.AutomaticSize.XY,
}),
})
end

return RobloxProfile
27 changes: 27 additions & 0 deletions src/About/RobloxProfile.story.luau
Original file line number Diff line number Diff line change
@@ -0,0 +1,27 @@
local React = require("@pkg/React")

local ContextProviders = require("@root/Common/ContextProviders")
local MockPlugin = require("@root/Testing/MockPlugin")
local RobloxProfile = require("./RobloxProfile")

local controls = {
userId = 1,
}

type Props = {
controls: typeof(controls),
}
return {
controls = controls,
story = function(props: Props)
local userId = assert(tonumber(props.controls.userId))

return React.createElement(ContextProviders, {
plugin = MockPlugin.new(),
}, {
RobloxProfiler = React.createElement(RobloxProfile, {
userId = userId,
}),
})
end,
}
15 changes: 15 additions & 0 deletions src/Common/useDescendants.luau
Original file line number Diff line number Diff line change
@@ -1,13 +1,24 @@
local React = require("@pkg/React")
local Sift = require("@pkg/Sift")

local function hasPermission(instance: Instance)
local success = pcall(function()
return instance.Name
end)
return success
end

local function useDescendants(parent: Instance, predicate: (descendant: Instance) -> boolean): { Instance }
local descendants: { Instance }, setDescendants = React.useState({})

local onDescendantChanged = React.useCallback(function(descendant: Instance)
setDescendants(function(prev)
local exists = table.find(prev, descendant)

if not hasPermission(descendant) then
return prev
end

if predicate(descendant) then
if exists then
-- Force a re-render. Nothing about the state changed, but the
Expand Down Expand Up @@ -41,6 +52,10 @@ local function useDescendants(parent: Instance, predicate: (descendant: Instance

-- Listen for name changes and update the list of descendants
for _, descendant in parent:GetDescendants() do
if not hasPermission(descendant) then
continue
end

table.insert(
connections,
descendant:GetPropertyChangedSignal("Name"):Connect(function()
Expand Down
8 changes: 6 additions & 2 deletions src/Forms/InputField.luau
Original file line number Diff line number Diff line change
Expand Up @@ -27,7 +27,7 @@ local function InputField(providedProps: Props)

local ref = React.useRef(nil :: TextBox?)
local text, setText = React.useState("")
local isValid, setIsValid = React.useState(false)
local isValid, setIsValid = React.useState(true)
local theme = useTheme()

local onFocusLost = React.useCallback(
Expand All @@ -41,15 +41,19 @@ local function InputField(providedProps: Props)
end
end,
{
text,
isValid,
props.onFocusLost,
props.onSubmit,
} :: { unknown }
)

local onTextChanged = React.useCallback(function(rbx: TextBox)
local newText = rbx.Text

if newText == text and newText ~= "" then
newText = newText:gsub("$%s+", ""):gsub("%s+^", "")

if newText == text or newText == "" then
return
end

Expand Down
1 change: 1 addition & 0 deletions src/Navigation/NavigationContext.luau
Original file line number Diff line number Diff line change
Expand Up @@ -13,6 +13,7 @@ export type NavigationContext = {
navigateTo: (newScreen: Screen) -> (),
goBack: () -> (),
getBreadcrumbs: () -> { string },
canGoBack: () -> boolean,
currentScreen: Screen,
}

Expand Down
Loading

0 comments on commit e8d4eb2

Please sign in to comment.