21
\$\begingroup\$

Given the following input:

  • An integer n where n > 0.
  • A string s where s is not empty and s~=[0-9A-Z]+ (alpha-numeric capitals only).

Using a standard, simplified QWERTY keyboard (as shown below):

1234567890
QWERTYUIOP
ASDFGHJKL
ZXCVBNM

Perform the following operation:

  • Find the original row that each character is in on the keyboard.
  • Replace the letter with the correct shifted equivalent for n based on its original position + n.
    • E.G. s="AB" and n=2: A would become D and B would become M.
  • If keyboard_row[position + n] > keyboard_row.length, wrap back to the start.
    • E.G. s="0P" and n=2: 0 would become 2 and P would become W.

Examples:

f("0PLM",1)    = 1QAZ
f("ZXCVB",2)   = CVBNM
f("HELLO",3)   = LYDDW
f("0PLM",11)   = 1QSV
f("0PLM",2130) = 0PHX

Rules


This is slightly more difficult than it seems at first glance.

\$\endgroup\$
2
  • 2
    \$\begingroup\$ Are we allowed to take the input as character-array instead of string? Currently assumed we, but forgot to ask.. \$\endgroup\$ Commented Feb 5, 2018 at 14:51
  • \$\begingroup\$ @KevinCruijssen shrug sure, it's not too outlandish. Unless it saves you a byte to break a tie I'm not complaining. \$\endgroup\$ Commented Feb 5, 2018 at 22:43

15 Answers 15

11
\$\begingroup\$

Jelly, 13 bytes

ØQØDṭ,ṙ€¥⁸F€y

Try it online!

How it works

ØQØDṭ,ṙ€¥⁸F€y  Main link. Left argument: n (integer). Right argument: s (string)

ØQ             Qwerty; set the return value to
               ["QWERTYUIOP", "ASDFGHJKL", "ZXCVBNM"].
  ØD           Digits; yield "0123456789".
    ṭ          Tack, yielding ["QWERTYUIOP", "ASDFGHJKL", "ZXCVBNM", "0123456789"].
        ¥⁸     Call the two links to the left as a dyadic chain, with right
               argument n.
      ṙ€       Rotate each string in the array n units to the left.
     ,         Yield the pair of the unmodified and the rotated string array.
          F€   Flatten each, mapping, e.g., ["QWERTYUIOP", ..., "0123456789"] to
               "QWERTYUIOPASDFGHJKLZXCVBNM0123456789".
            y  Translate s according to the mapping we've built.
\$\endgroup\$
5
  • 2
    \$\begingroup\$ Jelly has keyboard-layout built-ins eh? \$\endgroup\$ Commented Feb 5, 2018 at 14:13
  • 4
    \$\begingroup\$ @MagicOctopusUrn No, only QWERTY right now :-P \$\endgroup\$ Commented Feb 5, 2018 at 14:14
  • \$\begingroup\$ 13 bytes? Which character set is that supposed to be? In UTF-8 it's 26 bytes! \$\endgroup\$
    – Cephalopod
    Commented Feb 5, 2018 at 21:23
  • 2
    \$\begingroup\$ @Cephalopod Jelly uses the Jelly code page. \$\endgroup\$
    – Dennis
    Commented Feb 5, 2018 at 21:26
  • \$\begingroup\$ I'm not going to complain about regex anymore \$\endgroup\$
    – Xeoncross
    Commented Aug 22, 2023 at 14:52
9
\$\begingroup\$

Python 2, 110 bytes

lambda s,n,y='1234567890'*99+'QWERTYUIOP'*99+'ASDFGHJKL'*99+'ZXCVBNM'*99:''.join(y[y.find(c)+n%630]for c in s)

Try it online!

This uses a big enough string (99 copies of each row) and the LCM between the rows lengths (630) to find the correct substitution avoiding individual correction between each row.

\$\endgroup\$
7
\$\begingroup\$

Java 8, 159 158 bytes

n->s->{for(int i=s.length,j;i-->0;)for(String x:"1234567890;QWERTYUIOP;ASDFGHJKL;ZXCVBNM".split(";"))if((j=x.indexOf(s[i])+n)>=n)s[i]=x.charAt(j%x.length());}

-1 byte thanks to @OlivierGrégoire modifying the input-array instead of printing directly.

Explanation:

Try it online.

n->s->{  // Method with integer and character-array parameters, and no return-type
  for(int i=s.length,j;i-->0;)
         //  Loop over the input character-array with index
    for(String x:"1234567890;QWERTYUIOP;ASDFGHJKL;ZXCVBNM".split(";"))
         //   Inner loop over the qwerty-lines
      if((j=x.indexOf(s[i])+n)>=n)
         //    If the current qwerty-line contains the character
         //     Set `j` to the index of this character on that line + input `n`
        s[i]=x.charAt(j%x.length());}
         //     Replace the character at index `i`
         //     with the new character (at index `j` modulo length_of_qwerty_line)
\$\endgroup\$
1
  • 1
    \$\begingroup\$ 158 bytes, at the cost of input-outputting the char[]. \$\endgroup\$ Commented Feb 6, 2018 at 9:30
5
\$\begingroup\$

Retina, 49 bytes

"$&"+T`9o`dQW\ERTYUI\OPQASDFG\HJK\LAZXC\VBNMZ
0A`

Try it online! Takes input n and s on separate lines. Explanation:

"$&"+

Repeat n times.

T`9o`dQW\ERTYUI\OPQASDFG\HJK\LAZXC\VBNMZ

Shift all the characters one key to the right.

0A`

Delete n.

\$\endgroup\$
5
\$\begingroup\$

JavaScript (ES6), 101 99 bytes

Takes input in currying syntax (s)(n). Works with arrays of characters.

s=>n=>s.map(c=>(S='1QAZ2WSX3EDC4RFV5TGB6YHN7UJM8IK_9OL_0P')[(p=S.search(c)+n*4)%(-~'9986'[p%4]*4)])

Test cases

let f =

s=>n=>s.map(c=>(S='1QAZ2WSX3EDC4RFV5TGB6YHN7UJM8IK_9OL_0P')[(p=S.search(c)+n*4)%(-~'9986'[p%4]*4)])

console.log(JSON.stringify(f([..."0PLM"])(1)))    // 1QAZ
console.log(JSON.stringify(f([..."ZXCVB"])(2)))   // CVBNM
console.log(JSON.stringify(f([..."HELLO"])(3)))   // LYDDW
console.log(JSON.stringify(f([..."0PLM"])(11)))   // 1QSV
console.log(JSON.stringify(f([..."0PLM"])(2130))) // 0PHX

How?

We look for the position p of each character of the input within a string S where the keyboard rows are interleaved: the first 4 characters are '1QAZ' (first column of the keyboard), the next 4 characters are '2WSX' (second column of the keyboard) and so on. Unused positions are padded with underscores and the last ones are simply discarded.

col # | 0    | 1    | 2    | 3    | 4    | 5    | 6    | 7    | 8    | 9
------+------+------+------+------+------+------+------+------+------+---
row # | 0123 | 0123 | 0123 | 0123 | 0123 | 0123 | 0123 | 0123 | 0123 | 01
------+------+------+------+------+------+------+------+------+------+---
char. | 1QAZ | 2WSX | 3EDC | 4RFV | 5TGB | 6YHN | 7UJM | 8IK_ | 9OL_ | 0P

This allows us to easily identify the row with p mod 4 and eliminates the need for explicit separators between the rows.

We advance by 4n positions, apply the correct modulo for this row (40, 40, 36 and 28 respectively) and pick the replacement character found at this new position in S.

\$\endgroup\$
3
\$\begingroup\$

Jelly, 18 bytes

ØQØDṭẋ€‘}Ẏ©iЀ⁸+ị®

Try it online!

\$\endgroup\$
0
3
\$\begingroup\$

C,  152  149 bytes

Thanks to @gastropner for saving three bytes!

j,l;f(S,n){for(char*s=S,*k;*s;++s)for(k="1234567890\0QWERTYUIOP\0ASDFGHJKL\0ZXCVBNM\0";l=strlen(k);k+=l+1)for(j=l;j--;)k[j]-*s||putchar(k[(j+n)%l]);}

Try it online!

Unrolled:

j,l;
f(S,n)
{
    for (char*s=S, *k; *s; ++s)
        for (k="1234567890\0QWERTYUIOP\0ASDFGHJKL\0ZXCVBNM\0"; l=strlen(k); k+=l+1)
            for (j=l; j--;)
                k[j]-*s || putchar(k[(j+n)%l]);
}
\$\endgroup\$
2
  • \$\begingroup\$ Either I'm hallucinating, or the inner loop can be changed to for(j=l;j--;) but I don't know why without any other changed. Still, should get you to 149. \$\endgroup\$
    – gastropner
    Commented Feb 7, 2018 at 21:12
  • \$\begingroup\$ @gastropner Ah, yes, the search order doesn't matter, so it works. Thanks! \$\endgroup\$
    – Steadybox
    Commented Feb 7, 2018 at 22:11
2
\$\begingroup\$

Red, 152 bytes

f: func[s n][foreach c s[foreach[t l]["1234567890"10"QWERTYUIOP"10"ASDFGHJKL"9"ZXCVBNM"7][if p: find t c[if(i:(index? p)+ n // l)= 0[i: l]prin t/(i)]]]]

Try it online!

Ungolfed:

f: func [s n][1
    foreach c s [
        foreach [t l] ["1234567890"10"QWERTYUIOP"10"ASDFGHJKL"9"ZXCVBNM"7][
            p: find t c
            if p [ 
                i: (index? p) + n // l
                if i = 0 [i: l]
                prin t/(i) ]]]]
\$\endgroup\$
2
\$\begingroup\$

Haskell, 99 bytes

f(s,n)=[dropWhile(/=c)(cycle r)!!n|c<-s,r<-words"1234567890 QWERTYUIOP ASDFGHJKL ZXCVBNM",elem c r]

Try it online!

\$\endgroup\$
1
  • \$\begingroup\$ You can use s#n= ... instead of f(s,n)= ... which is only an example notation used for the examples. \$\endgroup\$
    – Laikoni
    Commented Feb 6, 2018 at 8:16
1
\$\begingroup\$

Perl 5, 94 + 1 (-p) = 95 bytes

$s=<>;for$i(1234567890,QWERTYUIOP,ASDFGHJKL,ZXCVBNM){eval"y/$i/".(substr$i,$s%length$i)."$i/"}

Try it online!

\$\endgroup\$
2
  • \$\begingroup\$ Damn, I didn't see your answer. They're basically the same, feel free to use my optimisations and I'll remove my answer. Let me know, if not, I'll just remove this comment :) \$\endgroup\$ Commented Feb 5, 2018 at 17:04
  • \$\begingroup\$ @DomHastings They are different enough. Please keep both. I like to see variations in approach. I learn from all of them... \$\endgroup\$
    – Ton Hospel
    Commented Feb 5, 2018 at 22:22
1
\$\begingroup\$

Japt, 20 bytes

Running out the door to dinner so more golfing and an explanation to follow.

;£=D·i9òs)æøX)gV+UbX

Try it

\$\endgroup\$
1
\$\begingroup\$

Perl, 59 58 57 56 bytes

Includes + for -p

Give input on STDIN as 2 lines, first the string, then the repeat

(echo 0PLM; echo 2130) | perl -pe '$a="OPQWERTYUILASDF-MZXCVBNM0-90";eval"y/HI$a/J$a/;"x<>'
\$\endgroup\$
1
  • \$\begingroup\$ Wow, I can't believe you got 29 bytes off mine! I was pretty happy with it originally... \$\endgroup\$ Commented Feb 9, 2018 at 8:14
0
\$\begingroup\$

Perl 5, 85 bytes

84 bytes code + 1 for -p.

eval join"",map"y/$_/@{[/./&&$'.$&]}/;",(1234567890,QWERTYUIOP,ASDFGHJKL,ZXCVBNM)x<>

Try it online!

\$\endgroup\$
0
\$\begingroup\$

Clean, 144 119 bytes

import StdEnv

\n s=[l.[(i+n)rem(size l)]\\c<-s,l<-["1234567890","QWERTYUIOP","ASDFGHJKL","ZXCVBNM"],i<-[0..]&j<-:l|j==c]

Try it online!

Lambda function with the signature Int ![Char] -> [Char]

\$\endgroup\$
0
\$\begingroup\$

Ruby, 101 bytes

->s,n{n.times{s.tr! '1234567890QWERTYUIOPASDFGHJKLZXCVBNM','2345678901WERTYUIOPQSDFGHJKLAXCVBNMZ'};s}

Try it online!

I'm honestly a little disappointed that I couldn't do better with 'cleverer' methods. The closest I got was along the lines of

a=%w{1234567890 QWERTYUIOP ASDFGHJKL ZXCVBNM}
b=a.map{|r|r[1..-1]<<r[0]}*''
a*=''
n.times{s.tr! a,b}

for a net gain of 7 characters.

\$\endgroup\$

Not the answer you're looking for? Browse other questions tagged or ask your own question.