Cette fonctionnalité se couple particulièrement bien avec celle-là, puisque le téléchargement du fichier permet d’avoir accès à la progression.

Concrètement, on s’occupe d’aller activer le download directement par AJAX, d’avoir le retour directement dans un blob puis de générer un bouton qu’on s’occupera de cliquer de manière seamless.

// La façon la plus simple, sans progression visuelle pour l'utilisateur
File.download('/export')
  .then((link) => { link.click() })

// Avec un paramètre X
File.download('/export', { x : 'paramètre' })
  .then((link) => { link.click() })

// Avec un paramètre et la progression. Je simule partiellement un component en VueJS pour les besoins de la cause :)
methods: {
  loadFile() {
    File.download('/export', { x : 'paramètre' }, this.progress)
      .then((link) => { link.click() })
  },
  progress(e) {
    // Attribuer les valeurs à notre component ProgressBar
    console.log('PROGRESS', e)
  }
}
import Vue from 'vue'

// Votre librairie favorite pour les calls HTTP
const client = {}

export default {
  download(url, data, progressCallBack) {
    let request = client.get(url, {
      progress: progressCallBack
    })
    return request.then((response) => {
      var result = document.createElement('a');
      var contentDisposition = response.headers.get('Content-Disposition') || '';
      var filename = contentDisposition.match(/filename=(.+);/)[1];
      filename = filename.replace(/"/g, '')
      return response.blob()
        .then(function (data) {
          result.href = window.URL.createObjectURL(data);
          result.target = '_self';
          result.download = filename;
          return result;
        })
    })
  }
}

Ce petit component est très utile pour faire afficher une barre de progression en respectant la reactivity“ de VueJS. Le code parle de lui-même, ce n’est même pas nécessaire de le décrire.

P.S Merci à Udy pour le progress natif :)

<template>
  <progress
    class="progress"
    :max="max"
    :value="percent"
    
    v-if="percent > 0"
  />
</template>

<script>
// Usage: <ProgressBar :percent="25" />

export default {
  name: 'progress-bar',
  props: {
    percent: {
      type: Number,
      required: false,
      default: 0
    },
    max: {
      type: Number,
      required: false,
      default: 100
    }
  }
}
</script>

<style lang="scss">
@import '~sass/tools';
.progress {
    position: relative;
    width: 100%;
    height: 10px;
    margin: 15px 0;
    color: cc(highlight);
    fill: cc(highlight);
    border: 1px solid cc(bg);
    &::-webkit-progress-bar {
        background-color: cc(bg);
        border: none;
        box-shadow: 0 0 0 1px cc(border) inset;
    }
    &::-webkit-progress-value {
        background-color: cc(highlight);
        border: none;
    }
}
</style>

Microsoft a ajouté le support du fameux .editorconfig dans Visual Studio depuis tous récemment. Avec la sortie de VS2017 hier en journée, pourquoi ne pas rafraîchir un peu notre environnement?

Le concept d’EditorConfig nous permet de garder les configurations du formatage directement dans le dépôt de code source. Cela devient donc extrêmement agréable de passer d’un projet à l’autre sans se soucier des normes de ceux-ci.

root = true

[*]
charset = utf-8
indent_style = space
indent_size = 4
insert_final_newline = true
trim_trailing_whitespace = true

[project.json]
indent_size = 2

# Front-end files
[*.{js,html,cshtml}]
indent_size = 2
end_of_line = lf

# C# files
[*.cs]
end_of_line = crlf

# New line preferences
csharp_new_line_before_open_brace = all
csharp_new_line_before_else = true
csharp_new_line_before_catch = true
csharp_new_line_before_finally = true
csharp_new_line_before_members_in_object_initializers = true
csharp_new_line_before_members_in_anonymous_types = true
csharp_new_line_within_query_expression_clauses = true

# Indentation preferences
csharp_indent_block_contents = true
csharp_indent_braces = false
csharp_indent_case_contents = true
csharp_indent_switch_labels = true
csharp_indent_labels = flush_left

# avoid this. unless absolutely necessary
dotnet_style_qualification_for_field = false:suggestion
dotnet_style_qualification_for_property = false:suggestion
dotnet_style_qualification_for_method = false:suggestion
dotnet_style_qualification_for_event = false:suggestion

# only use var when it's obvious what the variable type is
csharp_style_var_for_built_in_types = false:none
csharp_style_var_when_type_is_apparent = false:none
csharp_style_var_elsewhere = false:suggestion

# use language keywords instead of BCL types
dotnet_style_predefined_type_for_locals_parameters_members = true:suggestion
dotnet_style_predefined_type_for_member_access = true:suggestion

# name all constant fields using PascalCase
dotnet_naming_rule.constant_fields_should_be_pascal_case.severity = suggestion
dotnet_naming_rule.constant_fields_should_be_pascal_case.symbols  = constant_fields
dotnet_naming_rule.constant_fields_should_be_pascal_case.style    = pascal_case_style

dotnet_naming_symbols.constant_fields.applicable_kinds   = field
dotnet_naming_symbols.constant_fields.required_modifiers = const

dotnet_naming_style.pascal_case_style.capitalization = pascal_case

# static fields should have s_ prefix
dotnet_naming_rule.static_fields_should_have_prefix.severity = suggestion
dotnet_naming_rule.static_fields_should_have_prefix.symbols  = static_fields
dotnet_naming_rule.static_fields_should_have_prefix.style    = static_prefix_style

dotnet_naming_symbols.static_fields.applicable_kinds   = field
dotnet_naming_symbols.static_fields.required_modifiers = static

dotnet_naming_style.static_prefix_style.required_prefix = s_
dotnet_naming_style.static_prefix_style.capitalization = camel_case 

# internal and private fields should be _camelCase
dotnet_naming_rule.camel_case_for_private_internal_fields.severity = suggestion
dotnet_naming_rule.camel_case_for_private_internal_fields.symbols  = private_internal_fields
dotnet_naming_rule.camel_case_for_private_internal_fields.style    = camel_case_underscore_style

dotnet_naming_symbols.private_internal_fields.applicable_kinds = field
dotnet_naming_symbols.private_internal_fields.applicable_accessibilities = private, internal

dotnet_naming_style.camel_case_underscore_style.required_prefix = _
dotnet_naming_style.camel_case_underscore_style.capitalization = camel_case 

# Code style defaults
dotnet_sort_system_directives_first = true
csharp_preserve_single_line_blocks = true
csharp_preserve_single_line_statements = false

# Expression-level preferences
dotnet_style_object_initializer = true:suggestion
dotnet_style_collection_initializer = true:suggestion
dotnet_style_explicit_tuple_names = true:suggestion
dotnet_style_coalesce_expression = true:suggestion
dotnet_style_null_propagation = true:suggestion

# Expression-bodied members
csharp_style_expression_bodied_methods = false:none
csharp_style_expression_bodied_constructors = false:none
csharp_style_expression_bodied_operators = false:none
csharp_style_expression_bodied_properties = true:none
csharp_style_expression_bodied_indexers = true:none
csharp_style_expression_bodied_accessors = true:none

# Pattern matching
csharp_style_pattern_matching_over_is_with_cast_check = true:suggestion
csharp_style_pattern_matching_over_as_with_null_check = true:suggestion
csharp_style_inlined_variable_declaration = true:suggestion

# Null checking preferences
csharp_style_throw_expression = true:suggestion
csharp_style_conditional_delegate_call = true:suggestion

# Space preferences
csharp_space_after_cast = false
csharp_space_after_colon_in_inheritance_clause = true
csharp_space_after_comma = true
csharp_space_after_dot = false
csharp_space_after_keywords_in_control_flow_statements = true
csharp_space_after_semicolon_in_for_statement = true
csharp_space_around_binary_operators = before_and_after
csharp_space_around_declaration_statements = do_not_ignore
csharp_space_before_colon_in_inheritance_clause = true
csharp_space_before_comma = false
csharp_space_before_dot = false
csharp_space_before_open_square_brackets = false
csharp_space_before_semicolon_in_for_statement = false
csharp_space_between_empty_square_brackets = false
csharp_space_between_method_call_empty_parameter_list_parentheses = false
csharp_space_between_method_call_name_and_opening_parenthesis = false
csharp_space_between_method_call_parameter_list_parentheses = false
csharp_space_between_method_declaration_empty_parameter_list_parentheses = false
csharp_space_between_method_declaration_name_and_open_parenthesis = false
csharp_space_between_method_declaration_parameter_list_parentheses = false
csharp_space_between_parentheses = false
csharp_space_between_square_brackets = false

# Xml project files
[*.{csproj,vcxproj,vcxproj.filters,proj,nativeproj,locproj}]
indent_size = 2

# Shell scripts
[*.sh]
end_of_line = lf
[*.{cmd, bat}]
end_of_line = crlf

Il suffit de placer ce fichier directement à la source de votre projet, puis en redémarrant vos éditeurs (que ce soit Sublime, Atom, VSCode, etc.) les configurations prendront place.

Bon code!

P.S.Il est important de noter que les configurations d’un EditorConfig écrasent celles que vous avez manuellement placées dans votre éditeur.

Vous avez déjà entendu parler de SonarQube? Concrètement, c’est un analyseur de code source qui permet de calculer plusieurs métriques sur l’ensemble du codebase de votre projet. Il est disponible pour la majorité des gros langages et est incroyablement paramétrable. C’est tellement facile à utiliser que je ne trouve pas une bonne raison de ne pas le faire.

Pour ma part, je l’ai connecté directement dans AppVeyor. Ainsi, à chaque fois que j’ai un build qui est lancé, SonarQube est appelé afin d’en analyser son contenu.

Son intégration est simple, voici les 3 lignes d’instructions qui permettront de l’intégrer à votre projet:

init:
  - choco install "msbuild-sonarqube-runner" -y

before_build:
  - MSBuild.SonarQube.Runner.exe begin /k:"[UNIQUE_KEY]" /n:"[PROJECT_NAME]" /v:"[VERSION]" /d:sonar.host.url=https://sonarqube.com /d:sonar.login=[TOKEN]
  
after_build:
  - MSBuild.SonarQube.Runner.exe end /d:sonar.login=[TOKEN]

[UNIQUE_KEY]: Mettez ce que vous désirez. Cela doit être unique cependant, je conseil d’utiliser le nom de votre repository. (exemple: gabriel-robert1)

[PROJECT_NAME]: Mettez ce que vous désirez. C’est le nom qui sera affiché dans la liste des projets du côté de SonarQube. (exemple: Gabriel Robert)

[VERSION]: Mettez ce que vous désirez. (exemple: 1.0.0)

[TOKEN]: C’est la valeur du token que vous devez générer ici. Il serait préférable d’encrypter cette donnée avec les outils d’AppVeyor

That’s it!

Parfois on ne veut pas inclure certaines librairies plutôt lourde pour un besoin très minime. Dans mon cas, c’était une simple liste d’objets avec la possibilité d’en ajouter des occurences côté front-end. Je voulais éviter d’utiliser handlebars ou même pire, un framework complèt pour en venir à mes fins.

var _incrementIndex = function ($element) {
    var attributes = ["id", "name", "data-valmsg-for"];
    $element.find("input, span, select, textarea").each(function (index, elem) {
        
        var $elem = $(elem);

        for (index = 0; index < attributes.length; ++index) {
            var attrValue = attributes[index];
            if ($elem.attr(attrValue)) {
                var id = $elem.attr(attrValue).replace(/[^\d]/g, '');
                var newId = $elem.attr(attrValue).replace(id, parseInt(id, 10) + 1);
                $elem.attr(attrValue, newId);
            }
        }
    });
}

En passant une div à cette fonction, elle s’occupera de mettre à jour l’ensemble des index en gardant la schémantique de ASP .NET.