Skip to content

Instantly share code, notes, and snippets.

@nhoizey
Last active July 4, 2018 15:59
Show Gist options
  • Star 0 You must be signed in to star a gist
  • Fork 0 You must be signed in to fork a gist
  • Save nhoizey/224a1c9dfb396a4c7b41ea114f175712 to your computer and use it in GitHub Desktop.
Save nhoizey/224a1c9dfb396a4c7b41ea114f175712 to your computer and use it in GitHub Desktop.
Extracts the url of the largest image from a `srcset-w`
url = srcset.scan(/([^, ][^ ]+)\s+([0-9]+)w/).map{ |url, size| { 'url' => url.strip, 'size' => size.to_i } }.reduce({ 'url' => '', 'size' => 0 }){ |current, new| current = new if new['size'] > current['size'] }['url']

Source:


https://res.cloudinary.com/nho/image/fetch/c_limit,f_auto,q_auto,w_360/https://nicolas-hoizey.com/2003/10/gasteroprod_v1.png 360w,
https://res.cloudinary.com/nho/image/fetch/c_limit,f_auto,q_auto,w_670/https://nicolas-hoizey.com/2003/10/gasteroprod_v1.png 670w,
https://res.cloudinary.com/nho/image/fetch/c_limit,f_auto,q_auto,w_980/https://nicolas-hoizey.com/2003/10/gasteroprod_v1.png 980w,
https://res.cloudinary.com/nho/image/fetch/c_limit,f_auto,q_auto,w_1024/https://nicolas-hoizey.com/2003/10/gasteroprod_v1.png 1024w

Expected result: https://res.cloudinary.com/nho/image/fetch/c_limit,f_auto,q_auto,w_1024/https://nicolas-hoizey.com/2003/10/gasteroprod_v1.png


Source:

https://res.cloudinary.com/nho/image/fetch/c_limit,f_auto,q_auto,w_360/https://nicolas-hoizey.com/2002/10/doc-136.png 360w,
https://res.cloudinary.com/nho/image/fetch/c_limit,f_auto,q_auto,w_1024/https://nicolas-hoizey.com/2002/10/doc-136.png 1024w,
https://res.cloudinary.com/nho/image/fetch/c_limit,f_auto,q_auto,w_670/https://nicolas-hoizey.com/2002/10/doc-136.png 670w,
https://res.cloudinary.com/nho/image/fetch/c_limit,f_auto,q_auto,w_980/https://nicolas-hoizey.com/2002/10/doc-136.png 980w

Expected result: https://res.cloudinary.com/nho/image/fetch/c_limit,f_auto,q_auto,w_1024/https://nicolas-hoizey.com/2002/10/doc-136.png


Source:

https://example.com/some-small-image.jpg 320w, https://example.com/some-larger-image.jpg 640w, https://example.com/some-huge-image.jpg 2000w, https://example.com/some-large-image.jpg 980w

Expected result: https://example.com/some-huge-image.jpg

@tdd
Copy link

tdd commented Jul 4, 2018

Eeeeh voilà ça plante vu ton reduce incorrect.

@tdd
Copy link

tdd commented Jul 4, 2018

Mais au lieu de faire un map/reduce tu peux faire un reduce direct…

@tdd
Copy link

tdd commented Jul 4, 2018

@nhoizey v'là la bonne version :

def extract_largest_image(srcset)
  srcset
    .scan(/(\S+)\s+(\d+)w/)
    .map { |url, size| { url: url.strip, size: size.to_i } }
    .inject { |acc, cur| acc[:size] < cur[:size] ? cur : acc }[:url]
end

Explications :

  • Tu peux simplifier ta regex de scan en partant du principe que ton URL ne contient pas de whitespace : ton groupe URL se résume donc à \S+.
  • Préfère les symboles aux Strings pour tes hashes, c'est plus idiomatique la plupart du temps. La syntaxe x: y dans un contexte de hash équivaut à :x => y.
  • Si tu ne passes pas de valeur initiale à inject (nom "rubyesque" de reduce, qui en est l'alias), il prendra le premier élément comme accumulateur et démarrera l'itération au second, ce qui nous va parfaitement (idem en JS)
  • Dans un callback de réduction, le premier arg est l'accumulateur / résultat, le deuxième la valeur courante. On ne parle jamais de cur et new, sémantiquement ; en plus, new est mal vu comme identifiant car c'est normalement réservé à l'opérateur d'instantiation des classes (ex. Person.new )
  • L'itérateur d'un reduce doit toujours renvoyer l'accumulateur, même s'il n'a pas changé. C'est là où tu foirais ; le fait de réaffecter cur n'avait d'ailleurs aucun intérêt (référence bloc-locale), ce qui compte c'est la valeur de retour du bloc, donc de sa dernière expression évaluée, ici l'affectation (coup de bol dans un cas de série croissante)

Voilou…

@nhoizey
Copy link
Author

nhoizey commented Jul 4, 2018

Wow, c'est plus simple, plus clean, et super bien expliqué au point que j'ai tout compris… o_O

MERCI !!!!

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment