- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我正在尝试在 SwiftUI 中开发一款填字游戏应用。我想始终显示键盘,并能够选择一个正方形并让按键在那里输入一个字母。有没有办法始终在 View 中显示键盘?如果我使用 TextField,我可以让键盘显示出来,但是让拼图的每个单元格成为一个文本字段似乎相当笨拙。提前致谢。
最佳答案
最终产品:https://imgur.com/a/Q85GOWj
我决定删除我的其他答案并尝试练习代码。首先,您需要创建一个始终是第一响应者的 UITextField。您可以这样做:
struct PermanentKeyboard: UIViewRepresentable {
@Binding var text: String
class Coordinator: NSObject, UITextFieldDelegate {
var parent: PermanentKeyboard
init(_ parent: PermanentKeyboard) {
self.parent = parent
}
func textField(_ textField: UITextField, shouldChangeCharactersIn range: NSRange, replacementString string: String) -> Bool {
//Async to prevent updating state during view update
DispatchQueue.main.async {
if let last = string.last {
//Check if last character is alpha
if last.isLetter {
//Changes the binding to the last character of input, UPPERCASED
self.parent.text = String(last).uppercased()
}
//Allows backspace
} else if string == "" {
self.parent.text = ""
}
}
return false
}
}
func makeCoordinator() -> Coordinator {
Coordinator(self)
}
func makeUIView(context: Context) -> UITextField {
let textfield = UITextField()
textfield.delegate = context.coordinator
//Makes textfield invisible
textfield.tintColor = .clear
textfield.textColor = .clear
return textfield
}
func updateUIView(_ uiView: UITextField, context: Context) {
uiView.text = text
//Makes keyboard permanent
if !uiView.isFirstResponder {
uiView.becomeFirstResponder()
}
//Reduces space textfield takes up as much as possible
uiView.setContentHuggingPriority(.defaultHigh, for: .vertical)
uiView.setContentHuggingPriority(.defaultHigh, for: .horizontal)
}
}
它被包装在一个 UIViewRepresentable 中,这样我们就可以在我们的 SwiftUI View 中使用它。当我们的 View 可见并且它的键盘永远不会被关闭时,这个文本字段将自动聚焦。
现在我们必须使用实际的填字游戏制作我们的 SwiftUI View 。首先,我将在我们的 ContentView 中定义一个结构,它将用作填字游戏框:
struct CharacterBox: View {
@Binding var character: String
@Binding var selectedInput: Int
let index: Int
var isSelected: Bool {
index == selectedInput
}
var body: some View {
ZStack {
//Required for tap gesture to work
Rectangle()
.fill(Color.white)
//To signify selection
Rectangle()
.stroke(isSelected ? Color.orange : Color.black)
Text(character)
.foregroundColor(.black)
}
.frame(width: 50, height: 50)
.onTapGesture {
selectedInput = index
}
}
}
这个 CharacterBox 结构可以绑定(bind)到我们 ContentView 中的状态变量字符串,如果它与选定的输入变量匹配,那么它也会被不可见的文本字段修改。
到 ContentView。我们需要为每个要绑定(bind)的 CharacterBox 创建一个状态变量,但显然我们不能手动完成。因此,我们将创建一个数组,并让每个 CharacterBox 根据其索引绑定(bind)到一个字符串。
在此之前,我创建了这个函数来帮助我们用一些空字符串初始化一个数组。
func createEmptyStringArray(count: Int) -> [String] {
var array = [String]()
for _ in 0..<count {
array.append("")
}
return array
}
如果你愿意,你可以把它放在 ContentView 之外;它还可以让您稍后重用它。然后我们将要绑定(bind)的输入定义为一个数组,如下所示:
@State private var inputs = createEmptyStringArray(count: 3)
@State private var selectedInput = 0
selectedInput 是 inputs 数组中字符串的索引,不可见文本字段将绑定(bind)到该字符串。要为每个 CharacterBox 进行绑定(bind),我们可以将此函数放在我们的 ContentView 中:
func getInputBinding(index: Int) -> Binding<String> {
Binding<String>(
get: {
inputs[index]
},
set: {
inputs[index] = $0
}
)
}
请注意,因为您定义的每个 CharacterBox 的索引不应更改,所以可以使用函数来获取这样的 Binding。但是,对于我们的 selectedIndex,我们需要它在每次 selectedIndex 更改时更新绑定(bind),因此我们必须改用计算属性。像这样定义它:
var selectedInputBinding: Binding<String> {
Binding<String>(
get: {
inputs[selectedInput]
},
set: {
inputs[selectedInput] = $0
}
)
}
最后,把它们放在一起,这就是我们的 body :
var body: some View {
ZStack {
PermanentKeyboard(text: selectedInputBinding)
//Hides textfield and prevents user from selecting/pasting/copying
Rectangle()
.fill(Color.white)
.frame(maxWidth: .infinity, maxHeight: .infinity)
//Your crossword with bindings to inputs here; this is hard coded/manually created
VStack(spacing: 1) {
ForEach(0 ..< inputs.count) { index in
CharacterBox(character: getInputBinding(index: index), selectedInput: $selectedInput, index: index)
}
}
}
}
关于ios - 如何始终在 SwiftUI 中显示键盘,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/65545374/
当我使用路径文件上的快捷方式在文件之间移动时,似乎我不仅仅是在文件之间移动。 我使用>转到一个文件,在该文件中我更改光标的位置并执行某些操作,然后按 gf noremap 关于vim 通过快捷方式直
我正在尝试使用 Pong P. Chu 的书来学习 Verilog。我有一个关于如何评估和实现始终 block 的问题。作者代码中的风格让我感到困惑。 在此示例中,他编写了一个具有两个输出寄存器“y1
我正在尝试制作一个聊天应用程序,因此我需要它始终接收服务器信息。因此,当请求完成时,在: http.onreadystatechange=function(){ 我再次调用该函数,因此: reques
当您在 always block 敏感度列表中使用通配符 @* 时,我对什么被视为输入有点困惑。例如,在下面的示例中,哪些信号被解释为导致 always block 被重新评估的输入? 据我了解,cl
我有一个充当调试器的程序。我为线程设置了一个 hw bp,将 dr0 设置为我希望 bp 所在的地址,将 dr7 设置为 1,因为我希望 bp 在每次执行该地址时生成一个事件。 它有效,但现在的问题是
如何每次都以管理员身份在 Windows 上运行 git bash。 操作系统 - Windows 10 家庭版 64 位 最佳答案 我在 Google 上找到了这个结果: 将 Git Bash 设置
使用 accept() 时或 getpeername() , sockaddr_storage总是有 ss_family=AF_INET6 : struct sockaddr_storage addr
我在 Cordova 方面还有另一个问题。我想在 Cordova 7.1.0 中使用插件“cordova.custom.plugins.exitapp”和“cordova-plugins-printe
我试图让模块通过 ISE 12.4 中的语法检查,但它给了我一个我不明白的错误。首先是代码片段: parameter ROWBITS = 4; reg [ROWBITS-1:0] temp; genv
我正在使用Cordova开发适用于iOS的应用程序,其中包括地理位置功能(我使用官方插件https://github.com/apache/cordova-plugin-geolocation)。我在
我想知道是否有可能只在敏感列表中的多个信号一起变化时才执行 always block 。 例如,假设我有一个信号“in”和另一个“posedge clk”。我希望在两个信号都发生变化时执行 alway
我需要实现一种算法来访问数据库来检查最后一个元素,以便计算新的元素。当然,第一次这是不可能的,因为数据库是空的,我得到 IndexOutOfBoundsException) index 0 reque
我正在利用我在网上找到的画廊系统,根据鼠标图像的接近程度,它会按比例增长。 链接:Gallery 好吧,我调整了代码以响应(如您所见正在 build 中)并且没有明显的问题。我的问题在更改分辨率时开始
我正在创建一个 kiosk 应用程序,我想确保它无论如何始终位于其他 Windows 应用程序和 Windows 任务栏之上。 我已经阻止了 Windows 键盘命令(alt-tab 等),但仍有可能
我即将开始一个新的 React 项目,并尝试利用我以前的知识来创建一些关于我如何构建应用程序的规则。 有些事情我认为是真的: Redux 保存整个应用程序的“主要”数据 如果需要跨应用程序共享,Red
当你打开 VS Code 时,终端默认是在底部打开的。您可以单击该图标将其向右移动。我想知道是否有办法将右侧打开设置为默认值。 谢谢。 最佳答案 是的 - 在 v1.20 中引入了设置 workb
我有一个Events表,其中包含各种类型的事件。我只关心其中一种类型。因此,我编写的每个查询都以开头 Events.objects.filter(event_type="the_type").\
我在单例中创建了一个Timer,并且我一直在努力解决为什么Timer没有触发。我查看了这里的帖子,但没有找到我认为可以直接回答我的问题的帖子。 class ConnectionStateMonitor
我在 TableViewController 中显示了一组项目。它们在 TVC 中正确显示。下面的代码会继续,但它只会继续到我的 MKMapItem 数组的 indexPath 0,而不是被单击的单元
我的 VC 是这样的: var coins = 50 // coins override func viewDidLoad() { super.viewDidLoad() if(SKP
我是一名优秀的程序员,十分优秀!