Refinements Icon

# Refinements [![Gem Version](https://badge.fury.io/rb/refinements.svg)](http://badge.fury.io/rb/refinements) [![Code Climate Maintainability](https://api.codeclimate.com/v1/badges/98be62123a95d5ac32d2/maintainability)](https://codeclimate.com/github/bkuhlmann/refinements/maintainability) [![Code Climate Test Coverage](https://api.codeclimate.com/v1/badges/98be62123a95d5ac32d2/test_coverage)](https://codeclimate.com/github/bkuhlmann/refinements/test_coverage) [![Circle CI Status](https://circleci.com/gh/bkuhlmann/refinements.svg?style=svg)](https://circleci.com/gh/bkuhlmann/refinements) Provides additional enhancements (refinements) to core Ruby objects. ## Table of Contents - [Features](#features) - [Requirements](#requirements) - [Setup](#setup) - [Usage](#usage) - [Requires](#requires) - [Using](#using) - [Examples](#examples) - [Array](#array) - [Big Decimal](#big-decimal) - [File](#file) - [Hash](#hash) - [Pathname](#pathname) - [String](#string) - [Tests](#tests) - [Versioning](#versioning) - [Code of Conduct](#code-of-conduct) - [Contributions](#contributions) - [License](#license) - [History](#history) - [Credits](#credits) ## Features - Provides Array refinements: - `#compress` - Removes `nil` and empty values without modifying itself. - `#compress!` - Removes `nil` and empty values while modifying itself. - Provides BigDecimal refinements: - `#inspect` - Allows one to inspect a big decimal with numeric representation. - Provides File refinements: - `.rewrite` - When given a file path and a block, it provides the contents of the recently read file for manipulation and immediate writing back to the same file. - Provides Hash refinements: - `#except` - Answers new hash with given keys removed without modifying calling hash. - `#except!` - Answers new hash with given keys removed while modifying calling hash. - `#symbolize_keys` - Converts keys to symbols without modifying itself. - `#symbolize_keys!` - Converts keys to symbols while modifying itself. - `#deep_merge` - Merges deeply nested hashes together without modifying itself. - `#deep_merge!` - Merges deeply nested hashes together while modifying itself. - `#reverse_merge` - Merges calling hash into passed in hash without modifying calling hash. - `#reverse_merge!` - Merges calling hash into passed in hash while modifying calling hash. - Provides Pathname refinements: - `#rewrite` - When given a block, it provides the contents of the recently read file for manipulation and immediate writing back to the same file. - Provides String refinements: - `#first` - Answers first character of a string or first set of characters if given a number. - `#last` - Answers last character of a string or last set of characters if given a number. - `#blank?` - Answers `true`/`false` based on whether string is blank or not (i.e. ``, `\n`, `\t`, `\r`). - `#up` - Answers string with only first letter upcased. - `#down` - Answers string with only first letter downcased. - `#camelcase` - Answers a camelcased string. Example: "ThisIsCamelcase". - `#snakecase` - Answers a snakecased string. Example: "this_is_snakecase". - `#titleize` - Answers titleized string. Example: "This Is Titleized". - `#use` - Provides hash value computation, via a block, by using only the keys as arguments to the block. ## Requirements 1. [Ruby 2.6.x](https://www.ruby-lang.org). 1. A solid understanding of [Ruby refinements and lexical scope](https://www.youtube.com/watch?v=qXC9Gk4dCEw). ## Setup Type the following to install: gem install refinements Add the following to your Gemfile file: gem "refinements" ## Usage ### Requires If all refinements are not desired, add the following to your `Gemfile` instead: gem "refinements", require: false ...then require the specific refinement, as needed. Example: require "refinements/arrays" require "refinements/big_decimals" require "refinements/files" require "refinements/hashes" require "refinements/pathnames" require "refinements/strings" ### Using Much like including/extending a module, you'll need modify your object(s) to use the refinement(s): class Example using Refinements::Arrays using Refinements::BigDecimals using Refinements::Files using Refinements::Hashes using Refinements::Pathnames using Refinements::Strings end ### Examples The following sections demonstrate how each refinement enriches your objects with new capabilities. #### Array example = ["An", nil, "", "Example"] example.compress # => ["An", "Example"] example # => ["An", nil, "", "Example"] example = ["An", nil, "", "Example"] example.compress! # => ["An", "Example"] example # => ["An", "Example"] #### Big Decimal BigDecimal.new("5.0E-10").inspect # => "#" #### File File.rewrite("/test.txt") { |content| content.gsub "[placeholder]", "example" } #### Hash example = {a: 1, b: 2, c: 3} example.except :a, :b # => {c: 3} example # => {a: 1, b: 2, c: 3} example = {a: 1, b: 2, c: 3} example.except! :a, :b # => {c: 3} example # => {c: 3} example = {"a" => 1, "b" => 2} example.symbolize_keys # => {a: 1, b: 2} example # => {"a" => 1, "b" => 2} example = {"a" => 1, "b" => 2} example.symbolize_keys! # => {a: 1, b: 2} example # => {a: 1, b: 2} example = {a: 1, b: 2, c: 3} example.slice :a, :c # => {a: 1, c: 3} example # => {a: 1, b: 2, c: 3} example = {a: 1, b: 2, c: 3} example.slice! :a, :c # => {a: 1, c: 3} example # => {a: 1, c: 3} example = {a: "A", b: {one: "One", two: "Two"}} example.deep_merge b: {one: 1} # => {a: "A", b: {one: 1, two: "Two"}} example # => {a: "A", b: {one: "One", two: "Two"}} example = {a: "A", b: {one: "One", two: "Two"}} example.deep_merge! b: {one: 1} # => {a: "A", b: {one: 1, two: "Two"}} example # => {a: "A", b: {one: 1, two: "Two"}} example = {a: 1, b: 2} example.reverse_merge a: 0, c: 3 # => {a: 1, b: 2, c: 3} example # => {a: 1, b: 2} example = {a: 1, b: 2} example.reverse_merge! a: 0, c: 3 # => {a: 1, b: 2, c: 3} example # => {a: 1, b: 2, c: 3} example = {unit: "221B", street: "Baker Street", city: "London", country: "UK"} example.use { |unit, street| "#{unit} #{street}" } # => "221B Baker Street" #### Pathname Pathname("/test.txt").rewrite { |content| content.sub "[placeholder]", "example" } #### String "example".first # => "e" "example".first 4 # => "exam" "instant".last # => "t" "instant".last 3 # => "ant" " \n\t\r".blank? # => true "example".up # => "Example" "EXAMPLE".down # => "eXAMPLE" "this_is_an_example".camelcase # => "ThisIsAnExample" "ThisIsAnExample".snakecase # => "this_is_an_example" "ThisIsAnExample".titleize # => "This Is An Example" ## Tests To test, run: bundle exec rake ## Versioning Read [Semantic Versioning](https://semver.org) for details. Briefly, it means: - Major (X.y.z) - Incremented for any backwards incompatible public API changes. - Minor (x.Y.z) - Incremented for new, backwards compatible, public API enhancements/fixes. - Patch (x.y.Z) - Incremented for small, backwards compatible, bug fixes. ## Code of Conduct Please note that this project is released with a [CODE OF CONDUCT](CODE_OF_CONDUCT.md). By participating in this project you agree to abide by its terms. ## Contributions Read [CONTRIBUTING](CONTRIBUTING.md) for details. ## License Copyright 2015 [Alchemists](https://www.alchemists.io). Read [LICENSE](LICENSE.md) for details. ## History Read [CHANGES](CHANGES.md) for details. Built with [Gemsmith](https://github.com/bkuhlmann/gemsmith). ## Credits Developed by [Brooke Kuhlmann](https://www.alchemists.io) at [Alchemists](https://www.alchemists.io).