CSS Progress Circle [chiuso]


112

Ho cercato in questo sito Web per trovare barre di avanzamento, ma quelle che sono riuscito a trovare mostrano cerchi animati che vanno al 100% completo.

Vorrei che si fermasse a determinate percentuali come nello screenshot qui sotto. C'è un modo per farlo usando solo CSS?

Barre di avanzamento circolari


Non sto cercando uno script, sto cercando informazioni CSS3 su questo.
Adam GunShy ha detto l'

8
Non importa la formulazione, la domanda "Come fare Css Progress Circle?" è ancora valido. Penso che questo dovrebbe essere riaperto con una nuova formulazione, questo risultato è primo nelle ricerche e contiene risposte obsolete.
Ciantic

da quale sito web proviene lo screenshot?
morale

10
Questo è il risultato numero uno su Google per "indicatore di avanzamento circolare css". Peccato che la domanda sia chiusa.
Gopherkhan

Se stai usando MENO potresti essere interessato a cssscript.com/pure-css-circular-percentage-bar
jchook

Risposte:


119

Ho creato un tutorial su come farlo esattamente con CSS3 e la libreria LESS JavaScript. Puoi trovare il post del blog qui: https://medium.com/secoya-tech/a917b80c43f9

Ecco un jsFiddle del risultato finale. La percentuale viene impostata tramite l' data-progressattributo. Le modifiche vengono animate utilizzando transizioni CSS.

gif dell'indicatore di avanzamento radiale


3
Non avevo idea che potessi farlo con css. Bello.
Hobbes

4
Tuttavia, un notevole calo delle prestazioni .. rendendolo inutilizzabile per la mia app = [
Hobbes

2
Roba eccellente. Un problema minore in Firefox (utilizzando l'edizione per sviluppatori 41.0a2) crea spigoli vivi visibili durante la trasformazione. È facile vedere se imposti il ​​progresso a 90 e il tempo di transizione a 10 secondi. Per risolvere il problema basta aggiungere outline: 1px solid transparent;al .mask, .fill, .shadowgruppo.
luopio

5
@ Hobbes, non puoi, il poster mente. Questa risposta utilizza grandi quantità di Javascript tramite una libreria chiamata LESS.
GetFree

6
Ho creato una versione Sass, nel caso qualcuno volesse provarla: gist.github.com/digitalbreed/84a19db69244b22519e03550ba010a25
digitalbreed

78

Ho creato un violino usando solo CSS .

.wrapper {
  width: 100px; /* Set the size of the progress bar */
  height: 100px;
  position: absolute; /* Enable clipping */
  clip: rect(0px, 100px, 100px, 50px); /* Hide half of the progress bar */
}
/* Set the sizes of the elements that make up the progress bar */
.circle {
  width: 80px;
  height: 80px;
  border: 10px solid green;
  border-radius: 50px;
  position: absolute;
  clip: rect(0px, 50px, 100px, 0px);
}
/* Using the data attributes for the animation selectors. */
/* Base settings for all animated elements */
div[data-anim~=base] {
  -webkit-animation-iteration-count: 1;  /* Only run once */
  -webkit-animation-fill-mode: forwards; /* Hold the last keyframe */
  -webkit-animation-timing-function:linear; /* Linear animation */
}

.wrapper[data-anim~=wrapper] {
  -webkit-animation-duration: 0.01s; /* Complete keyframes asap */
  -webkit-animation-delay: 3s; /* Wait half of the animation */
  -webkit-animation-name: close-wrapper; /* Keyframes name */
}

.circle[data-anim~=left] {
  -webkit-animation-duration: 6s; /* Full animation time */
  -webkit-animation-name: left-spin;
}

.circle[data-anim~=right] {
  -webkit-animation-duration: 3s; /* Half animation time */
  -webkit-animation-name: right-spin;
}
/* Rotate the right side of the progress bar from 0 to 180 degrees */
@-webkit-keyframes right-spin {
  from {
    -webkit-transform: rotate(0deg);
  }
  to {
    -webkit-transform: rotate(180deg);
  }
}
/* Rotate the left side of the progress bar from 0 to 360 degrees */
@-webkit-keyframes left-spin {
  from {
    -webkit-transform: rotate(0deg);
  }
  to {
    -webkit-transform: rotate(360deg);
  }
}
/* Set the wrapper clip to auto, effectively removing the clip */
@-webkit-keyframes close-wrapper {
  to {
    clip: rect(auto, auto, auto, auto);
  }
}
<div class="wrapper" data-anim="base wrapper">
  <div class="circle" data-anim="base left"></div>
  <div class="circle" data-anim="base right"></div>
</div>

Controlla anche questo violino qui (solo CSS)

@import url(http://fonts.googleapis.com/css?family=Josefin+Sans:100,300,400);
    
.arc1 {
    width: 160px;
    height: 160px;
    background: #00a0db;
    -webkit-transform-origin: -31% 61%;
    margin-left: -30px;
    margin-top: 20px;
    -webkit-transform: translate(-54px,50px);
    -moz-transform: translate(-54px,50px);
    -o-transform: translate(-54px,50px);
}
.arc2 {
    width: 160px;
    height: 160px;
    background: #00a0db;
    -webkit-transform: skew(45deg,0deg);
    -moz-transform: skew(45deg,0deg);
    -o-transform: skew(45deg,0deg);
    margin-left: -180px;
    margin-top: -90px;
    position: absolute;
    -webkit-transition: all .5s linear;
    -moz-transition: all .5s linear;
    -o-transition: all .5s linear;
}

.arc-container:hover .arc2 {
    margin-left: -50px;
    -webkit-transform: skew(-20deg,0deg);
    -moz-transform: skew(-20deg,0deg);
    -o-transform: skew(-20deg,0deg);
}

.arc-wrapper {
    width: 150px;
    height: 150px;
    border-radius:150px;
    background: #424242;
    overflow:hidden;
    left: 50px;
    top: 50px;
    position: absolute;
}
.arc-hider {
    width: 150px;
    height: 150px;
    border-radius: 150px;
    border: 50px solid #e9e9e9;
    position:absolute;
    z-index:5;
    box-shadow:inset 0px 0px 20px rgba(0,0,0,0.7);
}

.arc-inset  {
    font-family: "Josefin Sans";
    font-weight: 100;
    position: absolute;
    font-size: 413px;
    margin-top: -64px;
    z-index: 5;
    left: 30px;
    line-height: 327px;
    height: 280px;
    -webkit-mask-image: -webkit-linear-gradient(top, rgba(0,0,0,1), rgba(0,0,0,0.2));
}
.arc-lowerInset {
    font-family: "Josefin Sans";
    font-weight: 100;
    position: absolute;
    font-size: 413px;
    margin-top: -64px;
    z-index: 5;
    left: 30px;
    line-height: 327px;
    height: 280px;
    color: white;
    -webkit-mask-image: -webkit-linear-gradient(top, rgba(0,0,0,0.2), rgba(0,0,0,1));
}
.arc-overlay {
    width: 100px;
    height: 100px;
    background-image: linear-gradient(bottom, rgb(217,217,217) 10%, rgb(245,245,245) 90%, rgb(253,253,253) 100%);
    background-image: -o-linear-gradient(bottom, rgb(217,217,217) 10%, rgb(245,245,245) 90%, rgb(253,253,253) 100%);
    background-image: -moz-linear-gradient(bottom, rgb(217,217,217) 10%, rgb(245,245,245) 90%, rgb(253,253,253) 100%);
    background-image: -webkit-linear-gradient(bottom, rgb(217,217,217) 10%, rgb(245,245,245) 90%, rgb(253,253,253) 100%);

    padding-left: 32px;
    box-sizing: border-box;
    -moz-box-sizing: border-box;
    line-height: 100px;
    font-family: sans-serif;
    font-weight: 400;
    text-shadow: 0 1px 0 #fff;
    font-size: 22px;
    border-radius: 100px;
    position: absolute;
    z-index: 5;
    top: 75px;
    left: 75px;
    box-shadow:0px 0px 20px rgba(0,0,0,0.7);
}
.arc-container {
    position: relative;
    background: #e9e9e9;
    height: 250px;
    width: 250px;
}
<div class="arc-container">
    <div class="arc-hider"></div>
    <div class="arc-inset">
        o
    </div>
    <div class="arc-lowerInset">
        o
    </div>
    <div class="arc-overlay">
        35%
    </div>
    <div class="arc-wrapper">
        <div class="arc2"></div>
        <div class="arc1"></div>
    </div>
</div>

O questa bellissima barra di avanzamento rotonda con HTML5, CSS3 e JavaScript.



@panos ho provato la tua prima soluzione. Ho bisogno che .circle border sia 6px invece di 10px. Ho ottenuto lo stesso ma raggiungendo il 50%. Dà un sussulto e ricomincia l'animazione. Basta provare
Santhosh Kumar

@ Santosh-kumar, devi cambiare anche altri valori
Panos Kal.

@panos Come cambierei il progresso delle prime soluzioni? potrebbe essere fatto con un elemento dati? Sono piuttosto nuovo nell'animazione
anthonytherockjohnson

1
Secondo MDN clip è ora deprecato.
jstaab

36

Che ne dici?

HTML

<div class="chart" id="graph" data-percent="88"></div>

Javascript

var el = document.getElementById('graph'); // get canvas

var options = {
    percent:  el.getAttribute('data-percent') || 25,
    size: el.getAttribute('data-size') || 220,
    lineWidth: el.getAttribute('data-line') || 15,
    rotate: el.getAttribute('data-rotate') || 0
}

var canvas = document.createElement('canvas');
var span = document.createElement('span');
span.textContent = options.percent + '%';

if (typeof(G_vmlCanvasManager) !== 'undefined') {
    G_vmlCanvasManager.initElement(canvas);
}

var ctx = canvas.getContext('2d');
canvas.width = canvas.height = options.size;

el.appendChild(span);
el.appendChild(canvas);

ctx.translate(options.size / 2, options.size / 2); // change center
ctx.rotate((-1 / 2 + options.rotate / 180) * Math.PI); // rotate -90 deg

//imd = ctx.getImageData(0, 0, 240, 240);
var radius = (options.size - options.lineWidth) / 2;

var drawCircle = function(color, lineWidth, percent) {
        percent = Math.min(Math.max(0, percent || 1), 1);
        ctx.beginPath();
        ctx.arc(0, 0, radius, 0, Math.PI * 2 * percent, false);
        ctx.strokeStyle = color;
        ctx.lineCap = 'round'; // butt, round or square
        ctx.lineWidth = lineWidth
        ctx.stroke();
};

drawCircle('#efefef', options.lineWidth, 100 / 100);
drawCircle('#555555', options.lineWidth, options.percent / 100);

e CSS

div {
    position:relative;
    margin:80px;
    width:220px; height:220px;
}
canvas {
    display: block;
    position:absolute;
    top:0;
    left:0;
}
span {
    color:#555;
    display:block;
    line-height:220px;
    text-align:center;
    width:220px;
    font-family:sans-serif;
    font-size:40px;
    font-weight:100;
    margin-left:5px;
}

http://jsfiddle.net/Aapn8/3410/

Il codice di base è stato preso da Simple PIE Chart http://rendro.github.io/easy-pie-chart/


Questa è stata la soluzione migliore per me (non anche jquery!).
Andy B

2
Anche per me. Ecco come animarlo: drawCircle ('# efefef', options.lineWidth, 100/100); var i = 0; var int = setInterval (function () {i ++; drawCircle ('# 555555', options.lineWidth, i / 100); span.textContent = i + "%"; if (i> = 100) {clearInterval (int);} }, 100);
marlar

1
Come impostare i colori sfumati sul cerchio?
yaniv14

jsFiddle non mostra una barra su Chrome.
Esamo

10

Un'altra soluzione pura basata su CSS che si basa su due elementi arrotondati ritagliati che ruoto per ottenere l'angolo giusto:

http://jsfiddle.net/maayan/byT76/

Questo è il CSS di base che lo abilita:

.clip1 {
    position:absolute;
    top:0;left:0;
    width:200px;
    height:200px;
    clip:rect(0px,200px,200px,100px);
}
.slice1 {
    position:absolute;
    width:200px;
    height:200px;
    clip:rect(0px,100px,200px,0px);
    -moz-border-radius:100px;
    -webkit-border-radius:100px; 
    border-radius:100px;
    background-color:#f7e5e1;
    border-color:#f7e5e1;
    -moz-transform:rotate(0);
    -webkit-transform:rotate(0);
    -o-transform:rotate(0);
    transform:rotate(0);
}

.clip2 
{
    position:absolute;
    top:0;left:0;
    width:200px;
    height:200px;
    clip:rect(0,100px,200px,0px);
}

.slice2
{
    position:absolute;
    width:200px;
    height:200px;
    clip:rect(0px,200px,200px,100px);
    -moz-border-radius:100px;
    -webkit-border-radius:100px; 
    border-radius:100px;
    background-color:#f7e5e1;
    border-color:#f7e5e1;
    -moz-transform:rotate(0);
    -webkit-transform:rotate(0);
    -o-transform:rotate(0);
    transform:rotate(0);
}

e js lo ruota come richiesto.

abbastanza facile da capire ..

Spero che aiuti, Maayan


1
All'interno di jQuery, non è necessario impostare tutto l' -vendor-prefixesinterno .css()♪ Usa solotransform: 'rotate(' + degree + 'deg)'
Roko C.Buljan

1
Questo è il più facile e pulisce, ho lavorato a partire dall'esempio di @Maayan e ho ottenuto questo: jsfiddle.net/g8z64Ler
lukart
Utilizzando il nostro sito, riconosci di aver letto e compreso le nostre Informativa sui cookie e Informativa sulla privacy.
Licensed under cc by-sa 3.0 with attribution required.