Введите свое имя через D-pad

32

Головоломка:

Рассмотрим консольную / портативную игру с D-Pad, где вы должны ввести название своего рода. Это появилось во многих старых играх до того, как использование QWERTY было популяризировано в консолях (например, я считаю, что Wii использует раскладку клавиатуры QWERTY для ввода). Как правило, экранная клавиатура выглядит с эффектом *:

По умолчанию:

0 1 2 3 4 5 6 7 8 9
A B C D E F G H I J
K L M N O P Q R S T
U V W X Y Z _ + ^ =

С переключенным корпусом:

0 1 2 3 4 5 6 7 8 9
a b c d e f g h i j
k l m n o p q r s t
u v w x y z - + ^ =

То есть все буквенно-цифровые клавиши и следующее:

_: Один пробел
-: дефис
+: переключать регистр только на следующую букву: переключать регистр
^букв (то есть переключать регистр всех букв)
=: ввод, завершение

* Очевидно, я заменил ключи типа «BKSP» и «ENTER» на более короткие версии.

И тогда аппаратный будет включать в себя D-Pad (или какую - либо форму контроля , где вы могли бы пойти up, down, leftи right)

Экран также обычно позволяет перемещаться с одной стороны непосредственно на другую. То есть, если вы сфокусировались на букве J, нажатие rightпозволит вам перейти к букве A.

Всякий раз, когда я вводил свое имя, я всегда старался найти самый быстрый способ сделать это.

Цель:

Ваша программа будет принимать строковый ввод, который может включать любой буквенно-цифровой символ, включая пробел и дефис, и ваша цель - вывести кратчайшее количество нажатий клавиш на клавиатуре d для вывода требуемой строки.

Соображения:

Вам не нужно включать нажатой клавишу для нажатия фактического символа.
Фокус всегда начинается с A
Ввод =должен быть нажат в конце

Пример:

input: Code Golf
output: 43

Объяснил:
A -> C= 2
C-> ^= 6 (движется влево)
^-> o= 5
o-> d= 2
d-> e= 1
e-> += 5
+-> _= 1
_-> += 1
+-> G= 3
G-> o= 3
o-> l= 3
l-> f= 5
f-> == 6

Обратите внимание, что +для a _и a быстрее ударить дважды, Gчем для удара ^один раз, а затем поменять местами назад.

Представление победителя (я позволю по крайней мере 1 Вт) будет самым коротким решением (в байтах). Поскольку это мой первый вопрос, я надеюсь, что это понятно и не слишком сложно.

Tas
источник
12
Отличный вызов! Всего лишь одно очко, 48 часов, вероятно, слишком мало. Это то, сколько нужно времени, чтобы получить награды, поэтому должно быть больше около недели +.
Maltysen
@Maltysen спасибо за предложение, я обновил вызов
Tas
1
Можете ли вы обернуть его вертикально или горизонтально?
Алекс Рейкинг
2
@ AlexReinking, это отличный момент! Да, ты можешь.
Tas
Большой! Моя реализация делает это, поэтому я просто хотел перепроверить.
Алекс Рейкинг

Ответы:

5

Рубин (369 байт)

Принимает данные из командной строки.

K="0123456789"+('A'..'Z').to_a.join+" +^="
Q=K.downcase.sub' ','-'
def d x,y
t,s=(x/10-y/10).abs,(x%10-y%10).abs
[t,4-t].min+[s,10-s].min
end
def v s,i,l,a
return l if s.empty?
c,r=s[0],s[1..-1]
j=K.index(c.upcase)||36
return v(r,j,l+d(i,j),a)if a.include?c
s,p=d(i,37)+d(37,j),d(i,38)+d(38,j)
[v(r,j,l+s,a),v(r,j,l+p,a==K ? Q : K)].min
end
puts v("#{ARGV[0]}=",10,0,K)

Благодаря @Charlie :) сэкономил кучу байтов :)

Алекс Рейнинг
источник
j=(K.index(c.upcase) or 36)может быть заменен на, j=K.index(c.upcase)||36чтобы сохранить 4 байта. def d(x,y)можно заменить на, def d x,yчтобы сохранить байт, и то же самое для def v. v(...) ifчтобы v(...)ifдля другого байта. В последней строке v(...)можно заменить на, v ...чтобы сохранить 1 байт, и trueна, !!0чтобы сохранить другой байт.
Чарли
Благодарность! Я действительно не знаю Руби. Я перевел это с python ...
Алекс Рейкинг
Я также могу заменить &&на &и ||с |.
Алекс Рейкинг
Ваша первая строка ( K=...) может быть заменена на диапазон ( K='0123456789'+('A'..'Z').to_a.join+' +^=')
Чарли
Бреет еще 2 с!
Алекс Рейкинг
9

Swift 1.2, 812 588 670 байт

Редактировать: Убрал 224 байта, заменив большие массивы чисел на Range и преобразовав его вместо Array.

Edit2: добавлен цикл по вертикали

typealias S=String
typealias I=Int
var A:(I)->S={S(UnicodeScalar($0))},B:(I)->(I,I)={a in(a%10,a/10)},a=Array(48...57).map{A($0)},b=[a+(Array(65...90)+[32,43,94,61]).map{A($0)},a+(Array(97...122)+[45,43,94,61]).map{A($0)}],z=Process.arguments
z.removeAtIndex(0)
func C(e:I,f:I)->I{let(a,b)=B(e),(c,d)=B(f)
return min(abs(d-b), abs(4-(d-b)))+min(abs(c-a),abs(10-(c-a)))}
func D(c:S,_ e:I=10,_ f:Bool=false,_ g:Bool=false)->I{if count(c)==0{return C(e,39)}
let h=c.startIndex,i=c.endIndex,j=S(c[h])
if let k=find(b[f ?1:0],j){return C(e,k)+D(c[advance(h,1)..<i],k,(g ?(!f):f),false)}else{return min(C(e,37)+D(c,37,!f,true),C(e,38)+D(c,38,!f,false))}}
print(D(" ".join(z)))

Чтобы запустить, поместите код в .swiftфайл и запустите его сswift <filename> <your name>


При этом используется простой подход, при котором две «клавиатуры» хранятся в виде массивов.

B:(I)->(I,I)={a in(a%10,a/10)} Преобразует индекс из массива в позицию x, y на виртуальной клавиатуре.

func C(e:I,f:I)->I{let(a,b)=B(e),(c,d)=B(f) return abs(d-b)+min(abs(c-a),abs(10-(c-a)))} Принимает начальный / конечный индекс и возвращает минимальное количество ходов, которое нужно получить от одного к другому (с учетом горизонтального переноса)

func D(c:S,_ e:I=10,_ f:Bool=false,_ g:Bool=false)->IЯвляется основной рекурсивной функцией, выполняющей большинство вычислений. Он вычисляет расстояние от текущей позиции до целевого характера, если дело не должно измениться, то он рассчитывает как на сдвиг и на колпачках блокировку методов и принимает наималейшие.

Бегущие swift codegolf.swift Code Golfотпечатки43

Дэвид Скрундз
источник
Необходимо учитывать вертикальную переноску.
Алекс Рейкинг
Обновлено также для учета вертикальной переноски.
Дэвид Скрундз
4

Python 679 661 619 602 589 576 539 520 496 482 байт

Запустите это, и он запросит ввод (без текста подсказки). Для ввода Code Golfон печатает 43.

a=input()+'=';b=0;c="0123456789abcdefghijklmnopqrstuvwxyz-+^=";d=0;e=[0,1];f='+';g='^';h=[i.isupper()or i==' 'for i in a];i=abs;p=lambda z:all([i==b for i in z]);q=0
def l(z):global s;k=c.index(z.lower().replace(' ','-'));s=[k%10,int(k/10)];m,n=s;return sum([min(i(m-e[0]),i(10-(m-e[0]))),min(i(n-e[1]),i(4-(n-e[1])))])
def o(z):global d,e;d+=l(z);e=s
for r in a:
 if p(h[q:q+3]):o(g);b^=1
 if p(h[q:q+2]):
  if l(f)<l(g):o(f)
  else:o(g);b^=1
 if p([h[q]]):o(f)
 o(r);q+=1
print(d)

Полная программа:

input = input() + '='
capsOn = False

keys = "0123456789abcdefghijklmnopqrstuvwxyz-+^="
totalKeys = 0
caret = [0, 1]

shiftKey = '+'
capsKey = '^'

cases = [char.isupper() or char == ' ' for char in input]

def locate(char):
    """
        Find the location of the char on the keyboard
        regardless of case
    """
    location = keys.find(char.replace(' ', '-').lower())
    return [location % 10, int(location / 10)]


def dist(key):
    """
        Calculate the min dist to a char
    """
    nx, ny = locate(key)
    return sum([min(abs(nx - caret[0]), abs(10 - (nx - caret[0]))), min(abs(ny - caret[1]), abs(4 - (ny - caret[1])))])


def moveTo(char):
    """
        Move the caret to the char, ignoring case and
        adds the dist to the tally
    """
    global totalKeys, caret
    totalKeys = totalKeys + dist(char)

    print(keys[caret[0] + caret[1] * 10], '->', char, '=', dist(char))

    caret = locate(char)

diffCase = lambda case: all([i == capsOn for i in case])

for ind, ch in enumerate(input):
    if diffCase(cases[ind:ind + 3]): # use caps
        moveTo(capsKey)
        capsOn ^= 1
    elif diffCase(cases[ind:ind + 2]): # use closest
        if dist(shiftKey) < dist(capsKey):
            moveTo(shiftKey)
        else:
            moveTo(capsKey)
            capsOn ^= 1
    elif diffCase([cases[ind]]): # use shift
        moveTo(shiftKey)

    moveTo(ch) # apply the move

print('Total:', totalKeys)

Расширенный вывод из полной программы:

Code Golf
a -> C = 2
c -> ^ = 6
^ -> o = 5
o -> d = 2
d -> e = 1
e -> + = 5
+ -> _ = 1
- -> + = 1
+ -> G = 3
g -> o = 3
o -> l = 3
l -> f = 5
f -> = = 6
Total: 43

Сохраненный байт благодаря @justin https://codegolf.stackexchange.com/a/18983/42736
4 @xnor https://codegolf.stackexchange.com/a/40791/42736 19 благодаря @Alex

Дж Аткин
источник
Любая помощь приветствуется, так как я все еще изучаю Python, и это мой первый гольф-код.
Дж Аткин
Вы можете использовать пробел вместо подчеркивания во внутренних таблицах.
Алекс Reinking
Я не думал об этом, спасибо;)
J Аткин
3

C 675 байт

Принимает входные данные из аргумента командной строки. Использует рекурсивную магистраль:

#define Y(_) (!isdigit(_)?!isalpha(_)?3:1+(toupper(_)-65)/10:0)
#define X(_) (!isdigit(_)?!isalpha(_)?_-32&&_-45?_-43?9-(_==94):7:6:(toupper(_)-5)%10:_-48)
x,y,z;char*s;main(a,_,p,q,r){a<2?s[_]?!isdigit(s[_])&&((s[_]-32&&!isupper(s[_]))||!a)&&((s[_]-45&&!islower(s[_]))||a)?q=x,r=y,main(3,43),p=z,x=X(43),y=Y(43),main(3,s[_]),p+=z,x=X(s[_]),y=Y(s[_]),main(a,_+1),p+=z,x=q,y=r,main(3,94),q=z,x=X(94),y=Y(94),main(3,s[_]),q+=z,x=X(s[_]),y=Y(s[_]),main(!a,_+1),q+=z,z=(q<p?q:p):(main(3,s[_]),q=z,x=X(s[_]),y=Y(s[_]),main(a,_+1),z+=q):(main(3,61)):(a<3?s=((char**)_)[1],x=0,y=1,main(1,0),printf("%d",z):(x=X(_)-x,y=Y(_)-y,x+=10*(x<0),y+=4*(y<0),z=(x>5?10-x:x)+(y>2?4-y:y)));}
LambdaBeta
источник