181 lines
5.3 KiB
C
181 lines
5.3 KiB
C
|
|
#include "ft_printf.h"
|
|
|
|
/*
|
|
** FT_FLAG_TRANSFORM :
|
|
** if i = flag_p(&s) | -precision is calculated before width,
|
|
** | on str, if < length(str), cuts it,
|
|
** print = ft_precision() | on nbr, if > length(nbr), add '0' before,
|
|
** | then add '-' if negatif
|
|
** | (if precision given with flag '0', it's
|
|
** | ignored and cuted from s)
|
|
** if i = flag_w(s) | -width is caculated
|
|
** if flag_-(&s) | -if flag '-', rm '-' and width from s
|
|
** print = ft_rpadd() | -put extra width as ' ' to right,
|
|
** else if flag_0(&s) | -if flag '0', rm '0' and width from s
|
|
** print = ft_lpadd() | -put extra width as '0' to left
|
|
** else | -if just width
|
|
** print = ft_lpadd() | put extra width as ' ' to left
|
|
** // if flag_+(s) |
|
|
** // else if flag_space(s) |
|
|
** // if flag_'(s) |
|
|
** // if flag_#(s) |
|
|
** // print = ft_altfrm() |
|
|
**
|
|
** int flag_p(char **s);
|
|
** char *ft_precision(int i, char *print);
|
|
** int flag_w(char *s);
|
|
** void flag_-(char **s);
|
|
** char *ft_rpadd(int i, char *print);
|
|
** char *ft_lpadd(int i, char *print, char c);
|
|
*/
|
|
|
|
char *ft_precision(char *s, char *print)
|
|
{ char *test;
|
|
char *tmp;
|
|
int i; test = s;
|
|
|
|
if ((tmp = ft_strchr(s, '.')))
|
|
{ ft_putstr("(");ft_putstr(s);ft_putstr(")");
|
|
i = ft_atoi(tmp + 1);
|
|
while (*tmp)
|
|
*(tmp++) = '\0';
|
|
while (ft_strchr("#- +'0", *(++s)))
|
|
if (*s == '0')
|
|
*s = '.';
|
|
// tmp = s;
|
|
// while (ft_strchr("%#- +'0", *s))
|
|
// {
|
|
// if (*s == '0')
|
|
// s++;
|
|
// *tmp = *s;
|
|
// tmp++;
|
|
// s++;
|
|
// }
|
|
// while (*s)
|
|
// {
|
|
// *tmp = *s;
|
|
// s++;
|
|
// tmp++;
|
|
// }
|
|
/* *tmp = '\0';*/ s = test;ft_putstr("[");ft_putnbr(i);ft_putstr("]");ft_putstr("(");ft_putstr(s);ft_putstr(")");
|
|
// return (precision_trsf);
|
|
}
|
|
return (print);
|
|
}
|
|
|
|
char *ft_flag_transform(char *s, char *print)
|
|
{
|
|
// printf("|%i| - |%.0i| - |%.0i| - |%.i|\n",0,0,1,0);
|
|
// printf("|%i| - |%i| - |%.4i| - |%.4i|\n",10,-10,10,-10);
|
|
// printf("|%i| - |%05i| - |%05.4i| - |%.4i|\n",10,10,10,10);
|
|
// printf("|%i| - |%-5i| - |%-5.4i| - |%-.4i|\n",10,10,10,10);
|
|
// char * t = "trois";
|
|
// printf("|%s| - |%.0s| - |%.2s| - |%.10s|\n",t,t,t,t);
|
|
// char y = 'y';
|
|
// printf("|%c| - |%.0c| - |%.2c| - |%.10c|\n",y,y,y,y);
|
|
|
|
print = ft_precision(s, print);
|
|
// (print, s, type); // -regarde s'il y a un '.' suivit de
|
|
// chiffres, donc une precision
|
|
// -si oui transforme les chiffres en
|
|
// un int 'precision'
|
|
// -s'il y a un '.' sans chiffres,
|
|
// donner la valeur 0 a 'precision'
|
|
// -virer ces chiffres et le '.' de s
|
|
// -si flag '0', l'enlever de s aussi
|
|
// -transformer print en fonction de
|
|
// cette precision :
|
|
// -1 si type vaut s:
|
|
// -si length(s) > a precision
|
|
// raboter la fin de s pour afficher
|
|
// uniquement au max la longueur de
|
|
// precision
|
|
// -sinon laisser print tel quel
|
|
// -2 si type de s est "diouxX":
|
|
// -si length(s) < a precision
|
|
// ajouter des '0' avant le nombre
|
|
// mais apres le signe '-' si c'est
|
|
// un nombre negatif
|
|
// -3 si type de s est "aAeEfF":
|
|
// -non traite
|
|
// -4 si type de s est "gG":
|
|
// -non traite
|
|
// -5 sinon:
|
|
// -erreur
|
|
|
|
// int i;
|
|
//
|
|
// if ((i = flag_p(&s)))
|
|
// print = ft_precision(i, print);
|
|
// if ((i = flag_w(s)))
|
|
// {
|
|
// if (flag_-(&s))
|
|
// print = ft_rpadd(i, print);
|
|
// else if (flag_0(&s))
|
|
// print = ft_lpadd(i, print, '0');
|
|
// else
|
|
// print = ft_lpadd(i, print, ' ');
|
|
// }
|
|
|
|
// if (flag_+(s)) //
|
|
// else if (flag_space(s)) //
|
|
// if (flag_'(s)) //
|
|
// if (flag_#(s)) //
|
|
// print = ft_alternate_form(print) //
|
|
return (print);
|
|
}
|
|
|
|
/*
|
|
** FT_PRINTF :
|
|
** va_list ap;
|
|
** int length;
|
|
** char *print; | -contain the arg converted into a string
|
|
** char *type; | -contain the specifier type to use
|
|
** | by va_arg
|
|
** while s = next_word() | -return the next sequence to be print
|
|
** | (either a string, or a conversion)
|
|
** type = specifier(s) | -return the type if it's a conversion, or "%",
|
|
** | or NULL if it's a string. if it's a
|
|
** | single '%' it's considered as a string
|
|
** | if convers0, rmvs length & specifier from s
|
|
** if !type: ft_put_word() | -print the string if it wasn't a conversion
|
|
** else | or if it was a '%%'
|
|
** while strchr(s,*) | -for each * present, expand it into s in
|
|
** ft_expand_star()| order it appears
|
|
** print = ft_convert()| -convert the arg with its type,
|
|
** | then trsfm it into a str and rtrn the str
|
|
** ft_flag_transform() | -proceed all modification according to flags
|
|
** ft_put_word(print) | -print the string fully converted
|
|
** return (length) | -return the length of what was printed
|
|
*/
|
|
|
|
int ft_printf(char *string, ...)
|
|
{
|
|
char *s;
|
|
char *print;
|
|
char *type;
|
|
int length;
|
|
va_list ap;
|
|
|
|
length = 0;
|
|
va_start(ap, string);
|
|
while ((s = next_word(&string)) != NULL)
|
|
{
|
|
if ((type = specifier(s)) == NULL)
|
|
length += ft_put_word(s);
|
|
else
|
|
{
|
|
while (ft_strchr(s, '*'))
|
|
if (!(ft_expand_star(va_arg(ap, int), &s)))
|
|
return (-1);
|
|
if (!(print = ft_convert(ap, type)))
|
|
return (-1);
|
|
if (!(print = ft_flag_transform(s, print)))
|
|
return (-1);
|
|
length += ft_put_word(print);
|
|
}
|
|
}
|
|
return (length);
|
|
}
|