From 6f38538dea86efd42fea2a145c55c1fb12a25f27 Mon Sep 17 00:00:00 2001 From: Alejandro Celaya Date: Thu, 9 Aug 2018 19:50:22 +0200 Subject: [PATCH 01/24] Improved responsiveness in visits page --- src/short-urls/ShortUrlVisits.js | 59 +++++++++++++++--------------- src/short-urls/ShortUrlVisits.scss | 6 ++- 2 files changed, 35 insertions(+), 30 deletions(-) diff --git a/src/short-urls/ShortUrlVisits.js b/src/short-urls/ShortUrlVisits.js index b1c6cb66..98581361 100644 --- a/src/short-urls/ShortUrlVisits.js +++ b/src/short-urls/ShortUrlVisits.js @@ -1,15 +1,15 @@ -import preloader from '@fortawesome/fontawesome-free-solid/faCircleNotch'; -import FontAwesomeIcon from '@fortawesome/react-fontawesome'; -import { isEmpty, mapObjIndexed, pick } from 'ramda'; -import React from 'react'; -import { Doughnut, HorizontalBar } from 'react-chartjs-2'; -import Moment from 'react-moment'; -import { connect } from 'react-redux'; -import { Card, CardBody, CardHeader, UncontrolledTooltip } from 'reactstrap'; -import DateInput from '../common/DateInput'; -import VisitsParser from '../visits/services/VisitsParser'; -import { getShortUrlVisits } from './reducers/shortUrlVisits'; -import './ShortUrlVisits.scss'; +import preloader from '@fortawesome/fontawesome-free-solid/faCircleNotch' +import FontAwesomeIcon from '@fortawesome/react-fontawesome' +import { isEmpty, mapObjIndexed, pick } from 'ramda' +import React from 'react' +import { Doughnut, HorizontalBar } from 'react-chartjs-2' +import Moment from 'react-moment' +import { connect } from 'react-redux' +import { Card, CardBody, CardHeader, UncontrolledTooltip } from 'reactstrap' +import DateInput from '../common/DateInput' +import VisitsParser from '../visits/services/VisitsParser' +import { getShortUrlVisits } from './reducers/shortUrlVisits' +import './ShortUrlVisits.scss' const MutedMessage = ({ children }) =>
@@ -144,23 +144,24 @@ export class ShortUrlsVisits extends React.Component { -
-
e.preventDefault()} className="form-inline mt-4 float-md-right"> - - this.setState({ startDate: date }, () => this.loadVisits())} - className="short-url-visits__date-input" - /> - this.setState({ endDate: date }, () => this.loadVisits())} - className="short-url-visits__date-input" - /> - -
+
+
+
+ this.setState({ startDate: date }, () => this.loadVisits())} + /> +
+
+ this.setState({ endDate: date }, () => this.loadVisits())} + className="short-url-visits__date-input" + /> +
+
diff --git a/src/short-urls/ShortUrlVisits.scss b/src/short-urls/ShortUrlVisits.scss index 4ebf855e..75aadecc 100644 --- a/src/short-urls/ShortUrlVisits.scss +++ b/src/short-urls/ShortUrlVisits.scss @@ -1,3 +1,7 @@ +@import '../utils/base'; + .short-url-visits__date-input { - margin-left: 10px; + @media(max-width: $smMax) { + margin-top: 0.5rem; + } } From 3821735a898a3028bd7d6da1146cd4bc34ef6772 Mon Sep 17 00:00:00 2001 From: Alejandro Celaya Date: Thu, 9 Aug 2018 20:13:46 +0200 Subject: [PATCH 02/24] Updated DateInput to be clearable --- src/common/DateInput.js | 8 ++++++-- src/common/DateInput.scss | 15 ++++++++++++++- src/short-urls/CreateShortUrl.js | 1 + src/short-urls/ShortUrlVisits.js | 2 ++ 4 files changed, 23 insertions(+), 3 deletions(-) diff --git a/src/common/DateInput.js b/src/common/DateInput.js index 7df47732..b246405c 100644 --- a/src/common/DateInput.js +++ b/src/common/DateInput.js @@ -3,6 +3,7 @@ import FontAwesomeIcon from '@fortawesome/react-fontawesome'; import React from 'react'; import DatePicker from 'react-datepicker'; import './DateInput.scss'; +import { isNil } from 'ramda'; export default class DateInput extends React.Component { constructor(props) { @@ -11,6 +12,9 @@ export default class DateInput extends React.Component { } render() { + const { isClearable, selected } = this.props; + const showCalendarIcon = !isClearable || isNil(selected); + return (
- this.inputRef.current.input.focus()} - /> + />}
); } diff --git a/src/common/DateInput.scss b/src/common/DateInput.scss index f9f41b71..ba4563d5 100644 --- a/src/common/DateInput.scss +++ b/src/common/DateInput.scss @@ -1,4 +1,5 @@ @import '../utils/mixins/vertical-align'; +@import '../utils/base'; .date-input-container { position: relative; @@ -11,6 +12,18 @@ .date-input-container__icon { @include vertical-align(); - right: 15px; + right: .75rem; cursor: pointer; } + +.react-datepicker__close-icon.react-datepicker__close-icon { + @include vertical-align(); + right: 0; +} + +.react-datepicker__close-icon.react-datepicker__close-icon::after { + right: .75rem; + line-height: 11px; + background-color: #333; + font-size: 14px; +} diff --git a/src/short-urls/CreateShortUrl.js b/src/short-urls/CreateShortUrl.js index 94f61c27..3f7782e4 100644 --- a/src/short-urls/CreateShortUrl.js +++ b/src/short-urls/CreateShortUrl.js @@ -48,6 +48,7 @@ export class CreateShortUrl extends React.Component { selected={this.state[id]} placeholderText={placeholder} onChange={date => this.setState({ [id]: date })} + isClearable {...props} />; const formatDate = date => isNil(date) ? date : date.format(); diff --git a/src/short-urls/ShortUrlVisits.js b/src/short-urls/ShortUrlVisits.js index 98581361..543714ce 100644 --- a/src/short-urls/ShortUrlVisits.js +++ b/src/short-urls/ShortUrlVisits.js @@ -150,6 +150,7 @@ export class ShortUrlsVisits extends React.Component { this.setState({ startDate: date }, () => this.loadVisits())} />
@@ -157,6 +158,7 @@ export class ShortUrlsVisits extends React.Component { this.setState({ endDate: date }, () => this.loadVisits())} className="short-url-visits__date-input" /> From f8372876d79236515ca65c2e06a30597ed4a7e80 Mon Sep 17 00:00:00 2001 From: Alejandro Celaya Date: Thu, 9 Aug 2018 20:28:31 +0200 Subject: [PATCH 03/24] Improved app icons quality --- public/icons/icon-128x128.png | Bin 0 -> 5338 bytes public/icons/icon-144x144.png | Bin 0 -> 6316 bytes public/icons/icon-152x152.png | Bin 0 -> 7071 bytes public/icons/icon-192x192.png | Bin 0 -> 7126 bytes public/icons/icon-384x384.png | Bin 0 -> 21477 bytes public/icons/icon-72x72.png | Bin 0 -> 3010 bytes public/icons/icon-96x96.png | Bin 0 -> 3411 bytes public/icons/shlink-128.png | Bin 6851 -> 0 bytes public/icons/shlink-16.png | Bin 690 -> 0 bytes public/icons/shlink-24.png | Bin 1065 -> 0 bytes public/icons/shlink-32.png | Bin 1551 -> 0 bytes public/icons/shlink-64.png | Bin 3284 -> 0 bytes public/manifest.json | 38 +++++++++++++++++++++------------- 13 files changed, 24 insertions(+), 14 deletions(-) create mode 100644 public/icons/icon-128x128.png create mode 100644 public/icons/icon-144x144.png create mode 100644 public/icons/icon-152x152.png create mode 100644 public/icons/icon-192x192.png create mode 100644 public/icons/icon-384x384.png create mode 100644 public/icons/icon-72x72.png create mode 100644 public/icons/icon-96x96.png delete mode 100644 public/icons/shlink-128.png delete mode 100644 public/icons/shlink-16.png delete mode 100644 public/icons/shlink-24.png delete mode 100644 public/icons/shlink-32.png delete mode 100644 public/icons/shlink-64.png diff --git a/public/icons/icon-128x128.png b/public/icons/icon-128x128.png new file mode 100644 index 0000000000000000000000000000000000000000..f30426eb7a365e8214c730a98795529d5ad38645 GIT binary patch literal 5338 zcmV<06ea74P)#nq~)OQ!xm^1%aTng1C%^iAg3-mYJN9F*@gr4vdNc zafwMhSq_lNOeV{j7>zke3<${L!a<`1C2lMNk+p$tsA_tv<<)!d&L6K?s_Qj$*V52k z?|Tjh4)yB2Tlaq7@Ba2HI195@9AU%!fy`VM5iTV@NJ_I-2=Vv8SYQC)#zAjD5;z9z z1bz!*LsB;mzkdHdZl)G9RR9UHyOlR&f@oJ6ba;; zuEfzv0JSUXED2D%tZonrJDtTZ97zC-SwFv~PM!Kw4L8wcz+`7$3I-maQcif_B_&`S zuL16I=A__o*Qi0RL5hSMfq*j~1&5%Nd`=NC&zXzDgb>VAu$(*&3KL@#V30Ep1&hH- zyGH1sFyT|2IVfzn^8Ek_=!V3U*1v_NeBe?9if|U7&k{gNNSczMM`7-`07EL=1U&-H zfQDv+w-3bFb3Ew~p|26ZFrnP9FmbS#FH9_AM94mZ{I#o#X-$~?>EkwD-5ujVIEhPj zn9$>}m_0VgLpPLl z`@2Dp!so9nVsN>OC)YQU(5167J%pTC0TTGTD>7UBvoU3;k3~0?V>;{Kdk7#U46AT4 ze`;w?1~hqykN>!#%#i}UNdi1IHY%cTe&sV(Q4 zVZNM@tzZZsB@8UhZ=J~Cvdr%Ohz$q?|p+bi*?7Qr0? zoM*C*HXC^zTx&w2WDqJ$8j=~8_!&@=nUV9&YfB48AIM4qG_>f%lKGpce#I&eOfT*B zHXwm}uPe#;xf5!`FFyV?C590@9p5^}0 zkpw^7R9IO_IpMh};m4brc=OYEj|}>*2|>OxIaM(kR(<%(XfilT*vS1kMr=$Vd6=dvFEYi2_V|V@bBJl zI=I0CrFn_|=vcezfUik3RS9{c#>$6QNBj%sv02A;2lguzGux=eM=?$n$p9Ykc?B zMzkE{89D^WAqjr|PIKPP$3O3jvtVVI)`TPWFDe3nr4zjLVH>~P(wvvL!p?e)XErtx z*QHa5d@&N>lqC4|d#(I>Yi})?O@xX1dpwW z5YrunuU8P@lq7iRqc&dHmYpQnS+B9=l?a;QD15z&0ARV5f3&HIRomM#`u>&!2^Opj z6YU`O4bzHQBy>sAaOwl02YLi`2NG~AP*tRI|FjSj2Y0_fbmuXRC)Y*TP=dv;M9|vB`)7yx`1Z{e_}yK*p_Ep!=z^7Dj<*^Gqcrw70i?9*^DB!~t{CX$ zo=L?FsmM+c?X1^WxT-y#58QcWc7dh^E5mF%oG1uM(4$*`W=P8X3SXX7!i+2`pT9Dgb(C@GoC=n%i}22YzE6T4A%Fma9>Lu4Mf~HGlAKNIuKFb3 zeYKHz(rWZDuD_dmuPg0T@a{*ig1i#I zlvZV{0?fNIXqyK9!#gWkx-P=zPvd=-1bNi~T#DdlclV>ZG`CG8KB!CZ=$gps;tdJR z9UtIZ*O#R!Z?UB0>(AE{ZZmDoJPBMq%*W%iD>{`N|Mtfd>^RmpB|+XCK|;X2Q%iEb z_&bhiEMA-1OVF(dt{LV_Tl}~#`R5lJINoA*YU2xF)1DZQt_~BoNe~eNespIgQ-*r! zi^)TIBS65da7~8xd}r8lAkM;7VcN9R1$C=1x?g%dz>C{jIo@iVH(Au@7Hm0?V9DA@ zr{2NGZmM9)kiJKPyb++trx40v_pn{{_Sw4Uy1mBCI6Q$6pEC zlEC8%>gYj;cheFlLkxdcePwg(7#Ztk`L2F$j7Emsh7}5e=(wQ{q?HvOfWWPIU_1 zkc_Ny^We;K4mar>Z#7SkMF`k)LT7(OXT~Uh$`};^Ny92Lmkt3smjp+f4L0nGviF3J z*Cl9-8f-n7;0JG5`G0Ypj1w_js^I-Pjo~4;ZI7VQ{XC5B@8PxGu}+^cH00))TSJVj zbTjjk0B`J#(HuV`johlB?xaptk;+1 z88}5A!PB=^a%n&N6tI`KxA9+ZHk0h2wYe0*vv-6T)BlX+U`n|6#pCRcBvY=PJjBc5 z*%i1t>{DO-RXyRT(W`O`FGyCfs#s>4-!1UCL{8JMniG=$d^y~y4`S~4AR{YXDN1Qn zlYIT^;*`ZdCEB08BSa{eI!y-~b=xlH{fpHUlhEpu;1Hl^e~^IIgvlc>N7z-LZW$yE z$(|F*(*Yt0j0m~7YeIT=F)85>AGQ%`Gt-=0c1DfgBT@)3s?yEqs*BrW*k_xeC3MN6 zH4#4PK!W${5-D%LlrVdoeOK|{{ew?2qUv*ZEw+wzb^OdW~OfX(3?Kc4lOy3%3me>YI%-#E{kr zs-}cTYOOZ(hUpNX$4Q_Wk|nQ1*mK-cHpwA^3-N6*~sDj0_FRoB>fpo;Rpc#_7rd0dQl`*T;jJ)U7Jwj<<&jWoTSe=5+Fkb zan1HN+Xa|7$jibT%hG)Qnju-XxtY2q{k&moRk709m%L&xDTY^9}*JYbGX4 zo_)=Jzn(QZz_+HAVsuC<=}u#6w`wJoELsz`J&I<|_}~RgD{~0YZl1lqH^yswVz!^( zJT}0R*%etu`MfTx_+*{U!5WuWd(xChhX6ecDPj5MCU!R@Z9l(mxR3j;FGY}D^4A|e zXtjOYuN!e;&MtNcaQ0RZH(9Xi1RvVu7lL(M&EnZ1%nTJa0el+PX^N-cU|^~1f@}j0 z0XiszQBAUFZJ6E1?Ipq0!@N8=GhH$4Dhv~jw;DFv(1l4WbqHYV8<0Hma+vK$wN7<{ z2d0;G`7q0lD77hN>rDDfhX5Bw5?I?on4R_ZB&Z$ZXUXjHZf7qKC|0SmaS=DgOk00~ zR~30pF>?rT4kQgLsiafIaD@MKOPa2{sZv_Rv0LcPpT<&h43dUqScRMC?&;6hrj*#8 zy5bO^2hbEVd1&Q{l%$fgar@;#mdvR*uVN@Agn}w}PfU*{9%<5vw4IRv))|Oip1mVP ziBI7R6N`9gW?5bs;1Hk-+d(`jd2G!&6vLrD7Tj1)z$4PCkWRh1RmCblnOB*XdjPQV zYDvA?oyc~!m*?S_H zv$Ksac&E_b2Dd8s-fbbqp5utxj`sYy(%zq3-^gowqPP^ns7g2AySvly^6+w|#;mNt7b1uH+FAz){<hPq_Q z+D3l=K`X7@I3l;9)nMVOFwb>38YgK;wjStOQ>#m>Jh*ej0GG3zStw8fmThig&5kHj zNBEdA%Fl3{w)DCw*>+gtwcSy+9Ej5rm$+SmxGwRv*R7ipO8s5ytW{mAW^_)nLx4QM zqY4h6)OlfBE6=~zimJe$Xt=)F4lC(_-YSJW|vbQP`P=GpQeOK zDA2vWdJ~6u*cVA=y%jhF$OBXbR2vVzDd97NygV^CM4&zG+@*?cFFwE}l^zV)hTBmB zdj|{?Muglfo?UTz@w35{p+1ZpxCI>o^Z+zNGNHzE+P9wzl=`jqxY8lO#gDQ5JuICQ zvh9tO4JHiqW}QXk5FpQx5@w9@pVz~e5wQk%FtcQULx3Kz41T_)nYZ`HbAInja?pLm zAwUmFfd^kY!Jqf#mIPHrDnXCp5TG}p$gA+g`Uuaxo4eD|OMD8YK8FCk$`t(Hcbd8P zcUJ2;-IQ5r@j;KsTDNc8ikTMtV9?blJ zONRg#+Dd5uJ84Lg?e!v`G$h%6Ogr5-U~rkrWz}BB_V+Nj%td99O0iFnG+=v{$Jse4 zVZFqPu)2;~r>vLw1jSwj-IRpevTH2hNPu4AOHjp`08>Us$!`@Q*a$?O=_oj~3BjvMEbcuF z{Lz_@g2Nx%T>a_O>FhAZ$95On6HFy4L>#%-Kmb5NPJqK=L8YxV&*010Pq`aBZA z%#HJF4$fLpXE~Rpl#Krg)KE7`AYKQGoO$Y#XqAZ19TC;hmhex&df*_Cbmp#a2uqZQNt<&F1MADXrNNXmu(5A{Rhn0>`C&Hw-a07*qoM6N<$f^H;5#Q*>R literal 0 HcmV?d00001 diff --git a/public/icons/icon-144x144.png b/public/icons/icon-144x144.png new file mode 100644 index 0000000000000000000000000000000000000000..c72619d4ef025bb5624ef7a7e8c80b367b19b0d7 GIT binary patch literal 6316 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 HWE%7z&RR?C literal 0 HcmV?d00001 diff --git a/public/icons/icon-384x384.png b/public/icons/icon-384x384.png new file mode 100644 index 0000000000000000000000000000000000000000..8ff8e04fc7980716acee26ab7236481b23b8123d GIT binary patch literal 21477 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 Ef;4=l&;S4c literal 0 HcmV?d00001 diff --git a/public/icons/icon-96x96.png b/public/icons/icon-96x96.png new file mode 100644 index 0000000000000000000000000000000000000000..b9c886397f0fcbb4b64a906073289cd9f49ace2a GIT binary patch literal 3411 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 literal 0 HcmV?d00001 diff --git a/public/icons/shlink-128.png b/public/icons/shlink-128.png deleted file mode 100644 index 1d82aa73016cdff313f0d93b7ccfb2b5ac1025de..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 6851 zcmV;!8a(BRP)Zb=!23CZUsb z()~KCJ81$mO#?O^5<}8S-b;XnBp4Db*??zbgJ(PPY{`;mOR9Oi=lyX@#wC4$cixC0mp3W`Jnod693W_Cy6@BdGBHIF#zs*RuWf^Q}xJjmaeR;0OcQnzs!wYpR!d3 z{A=~HvJ*>I979V~@`tYq6o8oJBvp?LXYun5GeG&}^*CPwv4kWXmxN*;|3u=FxFK`H z-h1pT3jA{M%7*Dx%ZHP&)S3}s>56*REH7j6%DPGgGIB2HVUv=%@9hn35KC6n>q78*U}@TL zFTt(}7LK;_ck>H6_29;k!Bgv+_^>vNA!!pia@KA34sf?vva)`ZfJ2$q`aGM$7iYM+ zXQDGZ=mTEd)XW=u0yz-$Xc%C$BH+FZVFFzfJUlCV&TpwbzcM?IdnY={Ee1X4f5E*9 zusD;a?iy?76JwoOf4HUy{$jeDsb$t&x!6M{V37h0OBbexc{DyZ#f_?T%?r47g@>lQ zDR!zkZ}f~+C_q`N|6@@Eciilt*wwYpnN+4TrA+VAJU}J`jIIeB7J6Om!jRl`vxC2#o8QLym!q6b;h|}6 zzBaD_w=PH+IRGSQk7Vl5?gha3QY+ai1`G+qi!9tf*@eS;?P>{<}|)ck-oKZgfrbUjP7TJye!wD+QP`M5i>rV-e-gCc3zPimTsjPPGAm5cEj5 zE)>fSfkagVUz+Le)MtEQx|=Zrt^Fb?Yy$vAKz&oZ+h4?x61!FDR1y|XNS2PVGaw6_ zS6`axCLA~V9{>Pi1{^vU?cN;;tHM)v7h%;po{7uk?9GPVG{}msDaif^07byQ)6s5! znQ?=y{P0tSIIZHk^MPnw^2u=yin7#KPz2a5g1(M`v;cVPK(I$%Xi}-pV~YyNw=3-i zfdo_qK0C>swf8V2gkr`O#QA;%0DyD81Uu@&J@QhwRoYp8dw#n?5Q#~aj&V?upS|+s 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 diff --git a/public/icons/shlink-16.png b/public/icons/shlink-16.png deleted file mode 100644 index ded45981199531b4231110dd3b55beb70553ffd3..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 690 zcmV;j0!{siP)u4AJ%jbK4yNcIILx~bgEqBh~CMMM!UBNCTxS{6iP zfso~&P@8C_8(kFQLJA^_p~8=u#8KzYIL_QVckXSWXu+r*IB<9u56|y8=M~A(<$JNP z8aW-;Whz@CZMpnsIlu$aKb~>NZZxY{SW>V(?!U>Bwp^4Hv^NF#U2s`1dI(Wdq$Hz; zRI#uGAQF_EN$9wq&-oUeXZ?-b={5-l1b%H7hFmLFnO(4VUv70p=xPmdyW0ey=y-TO zhl1J2#Ow4cbeA>cx{rwP-00rA)e#PPOeTN;3-1ee;FR$*?*M%Tn1f)Lv+ z#onB@TqeJjXfU>Kgmo57K=V##v(y5i6g z*U`1@kn zyTO*@(^RLBJHP5skwV#HcG1QN2(p`1T+b&Gl9mPx0#5+rFdTr9sT_@Uim0LTg|avN Y2c+iL7)2Z(Jpcdz07*qoM6N<$g0EjVDF6Tf diff --git a/public/icons/shlink-24.png b/public/icons/shlink-24.png deleted file mode 100644 index ea9251f03f66b38c96140ac9c6461c3508943772..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 1065 zcmV+^1lIeBP)K8Ay!yO0WpnHLJCD=N~~3}7K$0#&N$O{rayQ7?r||NO;eIii{I+x z{Cw~E-n{2~k7x}?s)56r6fG;uG_rZcXflr`xs^~r53oZA4r@ZR9NHel@0BP8y~7s! zPLCtS!c(F&00FcBXQZuYIq+l!m8OQF37ooMQ>~lE)z0E>TS1NALHhytRUt9yaON5yWkXk`d%gRxzEi7c4wh=OIw0VJo=*4reiG znM+b1GD-(f3SR#vPTpo->7g=>&z}z9F$B*(SXSizQZ&QQSJNc}h}LjK*$T?tk{Z8* z#??-?ZSWO0M5poueFX~7UAEZ!ZDL7s(gCD^q-E0=Nz-$A7AXWztoH&GllPBgc=JRY zLla8^SfF;7LvrAU2?jf3b=J{7>F&mHO?$*}LcagOhq!=+xc5{7LlX>)WiT{BdxMXxtv!RC4oTh@6vc5wa=loIXbnfy zQtc{dE9zDn9BdEJ{pSq-OyqdJ$>i;makk#)qoK;lGaaK$%-Sr=77_-Uw`bHOhzr1p`a&qh?jteo3OERv8ux|_%3Tr#W1=-2Sp%E_?&N<6 jr66xBl~+VPhG_l|xcpj!#=jX%00000NkvXXu0mjfOeN?n diff --git a/public/icons/shlink-32.png b/public/icons/shlink-32.png deleted file mode 100644 index 813c1b0be176183c50cd6249461a1afa35376bee..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 1551 zcmV+q2JrcbP)jO*%e47Rl6D7n~M5D&&IPsCFQRmY* z&P-;KL98K4BvJUFhKXMsCmHbt;tSFU2o|tb4aDBk2eiGn_x9d<&OPUBKitxCEhO~8 zN=~xRUhACw-)sHXT6>EX8+%OPLEtY^aGov&0IpKF9_LDWfHoJ}bl?F6n<75RjSB-T zZ}4ME!Tw_wA05uqmvtG#G@EiB@UjL8v0`ISchHpQKYmS=)ljSEe^1-#$e^*P{o{55DiEkT^8ZRmDMyRjhTdMt}X>! zm^994Q5Vq&nu4OE5P+1hY`&knuM7il$y}4?|2If?+L>0UBtXtqXEaEJH6nfqVG99W^Z%NKe7RUW-xt^nSc_uF0RSi4qS=lq&Cd!S6gan0FD=B&q~R zr5)b*W`yH87uTD-ldBv2JaJu=WLV?=tEwt=`+F=lex1RmPY;1!&Ux5&IM0#663ulc zenWEGr6JDH zJTL*Gx!xQf?>bUs!xw3?MQ_@RkB8u-XcBz7z^4oLrY!#T$q*-U?xc_Qbz7|4k*5DN zx)}keOnCV5c!@pTR>gE@zs=u28KyCz)4DK7ETExg06_nJkuh{Zl~1s;Da2h@gbA9W zV)6AmGrYPc$p!I=PdDuv;r;J(cpg&@hkokfA)jDPONcuzuOeVn{%K#z;-zgvw5>^! z3~6j>AE7_%a&uFVKQ50DG9_F0X0cs0rGhjLz;+d@E)21DX=Kt5c8#&UE~?X*FnD)Q zmN$2gph-dd(IQV>ALG8IVRQ-a?8#y|XAfc=fD|w^>QZu5MNaQNT4ckQ8Pa)|1&MMB zrpK5cFa$YUvEj=!MMrVZm0>)k*t$1|?W(Df(9i_Cj~3a|o*h>n=&@M8BhApLi>?Ve z`)xcARxAo~LEOL@i+RpcyzzC0kABE=?~*E(%=4Xf2($&aKKs{`FG9efUYqYy7T+JU z*xa6_Fjiy);3!CC99B1nXs9;$E>+Hux9_i|Z^UJLSAnJT{KNv1Z-2GUs!L8vLrp?Q zpUuaec^tRg0mlIXU;JF)wXZVNM|DD`)sr?gcn+4^k*H`ZBH|hEoal~UyZQyrvj^6LM*N|@w{0AaBAip=&|_s_F*orHF@r) zIPsvwam$1rwLRY3pQFkrsf+4(N=*udoM|%=0zT_1ux`gN&GjbF-WX#}SR2g+3c+Yyo`sV*$S*c=Fm9e{PNO&Ym36fZ*vH;vDX?865R^>6U7mY7L%# z{{$V!Y&1Db08aLj{~yXD5Zrlrn02=&fO1{gv@63Ubv~ME%a!!e|E z!1LgPj!_Qw+RTgVSS3Z)@>tg3r@79DQf0gO)dm5he8!wL04G5R;Fl96x&~}&t2F+$ zDp7Ha1JETjR2%#uM7OTpyVhH^;sM^W(_FC=0r&T<2O>uFKL&$M#2C9002ovPDHLkV1lu` B&VT>_ diff --git a/public/icons/shlink-64.png b/public/icons/shlink-64.png deleted file mode 100644 index 0dcd6f226bc9300757a44034de814d06eac82767..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 3284 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 SzShqG0000 Date: Fri, 10 Aug 2018 07:32:13 +0200 Subject: [PATCH 04/24] Fixed PWA name in manifest file --- public/manifest.json | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/public/manifest.json b/public/manifest.json index 098a4806..d0a49250 100644 --- a/public/manifest.json +++ b/public/manifest.json @@ -1,6 +1,6 @@ { "short_name": "Shlink", - "name": "Shlink web client", + "name": "Shlink", "start_url": "/", "display": "standalone", "theme_color": "#4696e5", From e4d5424c078fc02db2132209cf453ba19d9ff087 Mon Sep 17 00:00:00 2001 From: Alejandro Celaya Date: Fri, 10 Aug 2018 21:38:24 +0200 Subject: [PATCH 05/24] Fixed short URLs ordering in desktop resolutions --- src/short-urls/ShortUrlsList.js | 23 +++++++++++++++++------ src/short-urls/ShortUrlsList.scss | 4 ++++ src/short-urls/helpers/ShortUrlsRow.scss | 3 ++- 3 files changed, 23 insertions(+), 7 deletions(-) diff --git a/src/short-urls/ShortUrlsList.js b/src/short-urls/ShortUrlsList.js index 68afb2a3..b603b8ac 100644 --- a/src/short-urls/ShortUrlsList.js +++ b/src/short-urls/ShortUrlsList.js @@ -1,7 +1,7 @@ import caretDownIcon from '@fortawesome/fontawesome-free-solid/faCaretDown'; import caretUpIcon from '@fortawesome/fontawesome-free-solid/faCaretUp'; import FontAwesomeIcon from '@fortawesome/react-fontawesome'; -import { isEmpty, pick } from 'ramda'; +import { head, isEmpty, pick } from 'ramda'; import React from 'react'; import { connect } from 'react-redux'; import { ShortUrlsRow } from './helpers/ShortUrlsRow'; @@ -20,10 +20,10 @@ export class ShortUrlsList extends React.Component { constructor(props) { super(props); - const orderBy = props.shortUrlsListParams.orderBy; + const { orderBy } = props.shortUrlsListParams; this.state = { - orderField: orderBy ? Object.keys(orderBy)[0] : 'dateCreated', - orderDir: orderBy ? Object.values(orderBy)[0] : 'ASC', + orderField: orderBy ? head(Object.keys(orderBy)) : undefined, + orderDir: orderBy ? head(Object.values(orderBy)) : undefined, } } @@ -33,13 +33,24 @@ export class ShortUrlsList extends React.Component { } render() { + const determineOrderDir = field => { + if (this.state.orderField !== field) { + return 'ASC'; + } + + const newOrderMap = { + 'ASC': 'DESC', + 'DESC': undefined, + }; + return this.state.orderDir ? newOrderMap[this.state.orderDir] : 'ASC'; + } const orderBy = field => { - const newOrderDir = this.state.orderField !== field ? 'ASC' : (this.state.orderDir === 'DESC' ? 'ASC' : 'DESC'); + const newOrderDir = determineOrderDir(field); this.setState({ orderField: field, orderDir: newOrderDir }); this.refreshList({ orderBy: { [field]: newOrderDir } }) }; const renderOrderIcon = field => { - if (this.state.orderField !== field) { + if (this.state.orderField !== field || this.state.orderDir === undefined) { return null; } diff --git a/src/short-urls/ShortUrlsList.scss b/src/short-urls/ShortUrlsList.scss index df2cf01d..171705de 100644 --- a/src/short-urls/ShortUrlsList.scss +++ b/src/short-urls/ShortUrlsList.scss @@ -13,3 +13,7 @@ .short-urls-list__header-icon { margin-right: 5px; } + +.short-urls-list__header-cell--with-action { + cursor: pointer; +} diff --git a/src/short-urls/helpers/ShortUrlsRow.scss b/src/short-urls/helpers/ShortUrlsRow.scss index 4f6037bc..da6e8f86 100644 --- a/src/short-urls/helpers/ShortUrlsRow.scss +++ b/src/short-urls/helpers/ShortUrlsRow.scss @@ -27,10 +27,11 @@ &:last-child { position: absolute; - top: 3px; + top: 3.5px; right: .5rem; width: auto; padding: 0; + border: none; } } } From c80fea2877f60045690b78689695706aac38cfea Mon Sep 17 00:00:00 2001 From: Alejandro Celaya Date: Fri, 10 Aug 2018 22:16:50 +0200 Subject: [PATCH 06/24] Added ordering control to short URLs list in mobile resolutions --- src/short-urls/ShortUrlsList.js | 191 +++++++++++++++++------------- src/short-urls/ShortUrlsList.scss | 9 ++ 2 files changed, 119 insertions(+), 81 deletions(-) diff --git a/src/short-urls/ShortUrlsList.js b/src/short-urls/ShortUrlsList.js index b603b8ac..ef997589 100644 --- a/src/short-urls/ShortUrlsList.js +++ b/src/short-urls/ShortUrlsList.js @@ -1,12 +1,20 @@ -import caretDownIcon from '@fortawesome/fontawesome-free-solid/faCaretDown'; -import caretUpIcon from '@fortawesome/fontawesome-free-solid/faCaretUp'; -import FontAwesomeIcon from '@fortawesome/react-fontawesome'; -import { head, isEmpty, pick } from 'ramda'; -import React from 'react'; -import { connect } from 'react-redux'; -import { ShortUrlsRow } from './helpers/ShortUrlsRow'; -import { listShortUrls } from './reducers/shortUrlsList'; -import './ShortUrlsList.scss'; +import caretDownIcon from '@fortawesome/fontawesome-free-solid/faCaretDown' +import caretUpIcon from '@fortawesome/fontawesome-free-solid/faCaretUp' +import FontAwesomeIcon from '@fortawesome/react-fontawesome' +import { head, isEmpty, pick, toPairs } from 'ramda' +import React from 'react' +import { connect } from 'react-redux' +import { DropdownItem, DropdownMenu, DropdownToggle, UncontrolledDropdown } from 'reactstrap' +import { ShortUrlsRow } from './helpers/ShortUrlsRow' +import { listShortUrls } from './reducers/shortUrlsList' +import './ShortUrlsList.scss' + +const SORTABLE_FIELDS = { + dateCreated: 'Created at', + shortCode: 'Short URL', + originalUrl: 'Long URL', + visits: 'Visits', +}; export class ShortUrlsList extends React.Component { refreshList = extraParams => { @@ -16,6 +24,34 @@ export class ShortUrlsList extends React.Component { ...extraParams }); }; + determineOrderDir = field => { + if (this.state.orderField !== field) { + return 'ASC'; + } + + const newOrderMap = { + 'ASC': 'DESC', + 'DESC': undefined, + }; + return this.state.orderDir ? newOrderMap[this.state.orderDir] : 'ASC'; + } + orderBy = field => { + const newOrderDir = this.determineOrderDir(field); + this.setState({ orderField: newOrderDir !== undefined ? field : undefined, orderDir: newOrderDir }); + this.refreshList({ orderBy: { [field]: newOrderDir } }) + }; + renderOrderIcon = (field, className = 'short-urls-list__header-icon') => { + if (this.state.orderField !== field) { + return null; + } + + return ( + + ); + }; constructor(props) { super(props); @@ -32,78 +68,6 @@ export class ShortUrlsList extends React.Component { this.refreshList({ page: params.page }); } - render() { - const determineOrderDir = field => { - if (this.state.orderField !== field) { - return 'ASC'; - } - - const newOrderMap = { - 'ASC': 'DESC', - 'DESC': undefined, - }; - return this.state.orderDir ? newOrderMap[this.state.orderDir] : 'ASC'; - } - const orderBy = field => { - const newOrderDir = determineOrderDir(field); - this.setState({ orderField: field, orderDir: newOrderDir }); - this.refreshList({ orderBy: { [field]: newOrderDir } }) - }; - const renderOrderIcon = field => { - if (this.state.orderField !== field || this.state.orderDir === undefined) { - return null; - } - - return ( - - ); - }; - - return ( - - - - - - - - - - - - - {this.renderShortUrls()} - -
orderBy('dateCreated')} - > - {renderOrderIcon('dateCreated')} - Created at - orderBy('shortCode')} - > - {renderOrderIcon('shortCode')} - Short URL - orderBy('originalUrl')} - > - {renderOrderIcon('originalUrl')} - Long URL - Tags orderBy('visits')} - > - {renderOrderIcon('visits')} Visits -  
- ); - } - renderShortUrls() { const { shortUrlsList, selectedServer, loading, error, shortUrlsListParams } = this.props; if (error) { @@ -128,6 +92,71 @@ export class ShortUrlsList extends React.Component { /> )); } + + renderMobileOrderingControls() { + return ( +
+ + + Order by + + + {toPairs(SORTABLE_FIELDS).map(([key, value]) => + this.orderBy(key)}> + {value} + {this.renderOrderIcon(key, 'short-urls-list__header-icon--mobile')} + )} + + +
+ ); + } + + render() { + return ( + + {this.renderMobileOrderingControls()} + + + + + + + + + + + + + {this.renderShortUrls()} + +
this.orderBy('dateCreated')} + > + {this.renderOrderIcon('dateCreated')} + Created at + this.orderBy('shortCode')} + > + {this.renderOrderIcon('shortCode')} + Short URL + this.orderBy('originalUrl')} + > + {this.renderOrderIcon('originalUrl')} + Long URL + Tags this.orderBy('visits')} + > + {this.renderOrderIcon('visits')} Visits +  
+
+ ); + } } export default connect(pick(['selectedServer', 'shortUrlsListParams']), { listShortUrls })(ShortUrlsList); diff --git a/src/short-urls/ShortUrlsList.scss b/src/short-urls/ShortUrlsList.scss index 171705de..020081dd 100644 --- a/src/short-urls/ShortUrlsList.scss +++ b/src/short-urls/ShortUrlsList.scss @@ -14,6 +14,15 @@ margin-right: 5px; } +.short-urls-list__header-icon--mobile { + margin: 3.5px 0 0; + float: right; +} + .short-urls-list__header-cell--with-action { cursor: pointer; } + +.short-urls-list__order-dropdown { + width: 100%; +} From b3be7df890d282f6d1c5dcdedf92573ff3dfa2e8 Mon Sep 17 00:00:00 2001 From: Alejandro Celaya Date: Fri, 10 Aug 2018 22:27:50 +0200 Subject: [PATCH 07/24] Improved tags filtering control --- src/short-urls/SearchBar.js | 3 ++- src/short-urls/SearchBar.scss | 4 ++++ src/utils/Tag.scss | 4 ++-- 3 files changed, 8 insertions(+), 3 deletions(-) diff --git a/src/short-urls/SearchBar.js b/src/short-urls/SearchBar.js index 5feaf65a..d82ae9f2 100644 --- a/src/short-urls/SearchBar.js +++ b/src/short-urls/SearchBar.js @@ -1,4 +1,5 @@ import searchIcon from '@fortawesome/fontawesome-free-solid/faSearch'; +import tagsIcon from '@fortawesome/fontawesome-free-solid/faTags'; import FontAwesomeIcon from '@fortawesome/react-fontawesome'; import React from 'react'; import { connect } from 'react-redux'; @@ -41,7 +42,7 @@ export class SearchBar extends React.Component { {!isEmpty(selectedTags) && (

- Filtering by tags: +   {selectedTags.map(tag => Date: Sat, 11 Aug 2018 18:27:51 +0200 Subject: [PATCH 08/24] Improved badge color --- src/index.scss | 5 +++++ src/short-urls/ShortUrlVisits.js | 2 +- 2 files changed, 6 insertions(+), 1 deletion(-) diff --git a/src/index.scss b/src/index.scss index cb686abc..34331b19 100644 --- a/src/index.scss +++ b/src/index.scss @@ -26,3 +26,8 @@ padding: 30px 30px 30px 20px; } } + +.badge-main { + color: #fff; + background-color: $mainColor; +} diff --git a/src/short-urls/ShortUrlVisits.js b/src/short-urls/ShortUrlVisits.js index 543714ce..99e241d8 100644 --- a/src/short-urls/ShortUrlVisits.js +++ b/src/short-urls/ShortUrlVisits.js @@ -123,7 +123,7 @@ export class ShortUrlsVisits extends React.Component {

{ shortUrl.visitsCount && - Visits: {shortUrl.visitsCount} + Visits: {shortUrl.visitsCount} } Visit stats for {shortLink}

From e1008fcff16cd5fae763b5bd491e6e68a64a40ad Mon Sep 17 00:00:00 2001 From: Alejandro Celaya Date: Sat, 11 Aug 2018 21:39:27 +0200 Subject: [PATCH 09/24] Replaced tags input component by a simpler one --- package.json | 2 +- src/common/react-tagsinput.scss | 54 ++++++++++++++++++++++++++++++ src/index.js | 3 ++ src/short-urls/CreateShortUrl.js | 26 +++++--------- src/short-urls/CreateShortUrl.scss | 12 ------- yarn.lock | 6 ++-- 6 files changed, 69 insertions(+), 34 deletions(-) create mode 100644 src/common/react-tagsinput.scss diff --git a/package.json b/package.json index b89f28c7..bd8f411a 100644 --- a/package.json +++ b/package.json @@ -28,7 +28,7 @@ "react-moment": "^0.7.6", "react-redux": "^5.0.7", "react-router-dom": "^4.2.2", - "react-tag-autocomplete": "^5.5.1", + "react-tagsinput": "^3.19.0", "reactstrap": "^6.0.1", "redux": "^4.0.0", "redux-thunk": "^2.3.0", diff --git a/src/common/react-tagsinput.scss b/src/common/react-tagsinput.scss new file mode 100644 index 00000000..2bddbf33 --- /dev/null +++ b/src/common/react-tagsinput.scss @@ -0,0 +1,54 @@ +.react-tagsinput { + background-color: #fff; + border: 1px solid #ccc; + border-radius: .25rem; + overflow: hidden; + min-height: calc(2.6rem + 2px); + padding: 6px 0 0 6px; +} + +.react-tagsinput--focused { + border-color: #80bdff; + -webkit-box-shadow: 0 0 0 0.2rem rgba(0,123,255,.25); + box-shadow: 0 0 0 0.2rem rgba(0,123,255,.25); + -webkit-transition: border-color .15s ease-in-out,-webkit-box-shadow .15s ease-in-out; + transition: border-color .15s ease-in-out,-webkit-box-shadow .15s ease-in-out; + -o-transition: border-color .15s ease-in-out,box-shadow .15s ease-in-out; + transition: border-color .15s ease-in-out,box-shadow .15s ease-in-out; + transition: border-color .15s ease-in-out,box-shadow .15s ease-in-out,-webkit-box-shadow .15s ease-in-out; +} + +.react-tagsinput-tag { + font-size: 1rem; + background-color: #f1f1f1; + border-radius: 2px; + border: 1px solid #d1d1d1; + display: inline-block; + font-weight: 400; + margin: 0 5px 6px 0; + padding: 6px 8px; + line-height: 1; +} +.react-tagsinput-tag:hover { + border-color: #b1b1b1; +} + +.react-tagsinput-remove { + cursor: pointer; + font-weight: bold; + margin-left: 8px; +} + +.react-tagsinput-tag a::before { + content: "\2715"; + color: #aaa; +} + +.react-tagsinput-input { + background: transparent; + border: 0; + outline: none; + padding: 3px 5px; + width: 155px; + margin-bottom: 6px; +} diff --git a/src/index.js b/src/index.js index df2b7293..32082499 100644 --- a/src/index.js +++ b/src/index.js @@ -6,6 +6,9 @@ import { BrowserRouter } from 'react-router-dom'; import { applyMiddleware, compose, createStore } from 'redux'; import ReduxThunk from 'redux-thunk'; +import '../node_modules/react-datepicker/dist/react-datepicker.css'; +import './common/react-tagsinput.scss'; + import App from './App'; import './index.scss'; import ScrollToTop from './common/ScrollToTop' diff --git a/src/short-urls/CreateShortUrl.js b/src/short-urls/CreateShortUrl.js index 3f7782e4..4102a67b 100644 --- a/src/short-urls/CreateShortUrl.js +++ b/src/short-urls/CreateShortUrl.js @@ -1,12 +1,11 @@ import downIcon from '@fortawesome/fontawesome-free-solid/faAngleDoubleDown'; import upIcon from '@fortawesome/fontawesome-free-solid/faAngleDoubleUp'; import FontAwesomeIcon from '@fortawesome/react-fontawesome'; -import { assoc, dissoc, isNil, pick, pipe, pluck, replace } from 'ramda'; +import { assoc, dissoc, isNil, pick, pipe, replace, trim } from 'ramda'; import React from 'react'; import { connect } from 'react-redux'; -import ReactTags from 'react-tag-autocomplete'; +import TagsInput from 'react-tagsinput' import { Collapse } from 'reactstrap'; -import '../../node_modules/react-datepicker/dist/react-datepicker.css'; import DateInput from '../common/DateInput'; import './CreateShortUrl.scss'; import CreateShortUrlResult from './helpers/CreateShortUrlResult'; @@ -26,14 +25,7 @@ export class CreateShortUrl extends React.Component { render() { const { createShortUrl, shortUrlCreationResult, resetCreateShortUrl } = this.props; - const addTag = tag => this.setState({ - tags: [].concat(this.state.tags, assoc('name', replace(/ /g, '-', tag.name), tag)) - }); - const removeTag = i => { - const tags = this.state.tags.slice(0); - tags.splice(i, 1); - this.setState({ tags }); - }; + const changeTags = tags => this.setState({ tags: tags.map(pipe(trim, replace(/ /g, '-'))) }); const renderOptionalInput = (id, placeholder, type = 'text', props = {}) =>
-
diff --git a/src/short-urls/CreateShortUrl.scss b/src/short-urls/CreateShortUrl.scss index 7967eb77..731f63bd 100644 --- a/src/short-urls/CreateShortUrl.scss +++ b/src/short-urls/CreateShortUrl.scss @@ -6,18 +6,6 @@ margin-left: 5px; } -.react-tags { - @include border-radius(.25rem); - transition: border-color .15s ease-in-out, box-shadow .15s ease-in-out, -webkit-box-shadow .15s ease-in-out; -} -.react-tags.is-focused { - color: #495057; - background-color: #fff; - border-color: #80bdff; - outline: 0; - @include box-shadow(0 0 0 0.2rem rgba(0,123,255,.25)); -} - .react-datepicker__input-container, .react-datepicker-wrapper { display: block !important; diff --git a/yarn.lock b/yarn.lock index d9056d4c..8e47e890 100644 --- a/yarn.lock +++ b/yarn.lock @@ -6190,9 +6190,9 @@ react-router@^4.3.1: prop-types "^15.6.1" warning "^4.0.1" -react-tag-autocomplete@^5.5.1: - version "5.5.1" - resolved "https://registry.yarnpkg.com/react-tag-autocomplete/-/react-tag-autocomplete-5.5.1.tgz#6b3f253d3d69eb546925118cdf43138a9aafe113" +react-tagsinput@^3.19.0: + version "3.19.0" + resolved "https://registry.yarnpkg.com/react-tagsinput/-/react-tagsinput-3.19.0.tgz#6e3b45595f2d295d4657bf194491988f948caabf" react-test-renderer@^16.0.0-0: version "16.4.2" From c920403d5f1ee906f6dc48a6c2200ee4b1e73203 Mon Sep 17 00:00:00 2001 From: Alejandro Celaya Date: Sat, 11 Aug 2018 21:41:01 +0200 Subject: [PATCH 10/24] Deleted no longer needed styles sheet --- src/index.scss | 5 +++++ src/short-urls/CreateShortUrl.js | 1 - src/short-urls/CreateShortUrl.scss | 12 ------------ 3 files changed, 5 insertions(+), 13 deletions(-) delete mode 100644 src/short-urls/CreateShortUrl.scss diff --git a/src/index.scss b/src/index.scss index 34331b19..3cc74ea0 100644 --- a/src/index.scss +++ b/src/index.scss @@ -31,3 +31,8 @@ color: #fff; background-color: $mainColor; } + +.react-datepicker__input-container, +.react-datepicker-wrapper { + display: block !important; +} diff --git a/src/short-urls/CreateShortUrl.js b/src/short-urls/CreateShortUrl.js index 4102a67b..971a34f9 100644 --- a/src/short-urls/CreateShortUrl.js +++ b/src/short-urls/CreateShortUrl.js @@ -7,7 +7,6 @@ import { connect } from 'react-redux'; import TagsInput from 'react-tagsinput' import { Collapse } from 'reactstrap'; import DateInput from '../common/DateInput'; -import './CreateShortUrl.scss'; import CreateShortUrlResult from './helpers/CreateShortUrlResult'; import { createShortUrl, resetCreateShortUrl } from './reducers/shortUrlCreationResult'; diff --git a/src/short-urls/CreateShortUrl.scss b/src/short-urls/CreateShortUrl.scss deleted file mode 100644 index 731f63bd..00000000 --- a/src/short-urls/CreateShortUrl.scss +++ /dev/null @@ -1,12 +0,0 @@ -@import '../../node_modules/react-tag-autocomplete/example/styles.css'; -@import '../utils/mixins/box-shadow'; -@import '../utils/mixins/border-radius'; - -.create-short-url__btn:not(:first-child) { - margin-left: 5px; -} - -.react-datepicker__input-container, -.react-datepicker-wrapper { - display: block !important; -} From f9773dbebe0bf8791a0f40c5505cd374cfa7e6cd Mon Sep 17 00:00:00 2001 From: Alejandro Celaya Date: Sat, 11 Aug 2018 22:06:36 +0200 Subject: [PATCH 11/24] Added servers list to home page --- src/common/Home.js | 34 ++++++++++++++++++++++++++++------ src/common/Home.scss | 18 ++++++++++++++++++ 2 files changed, 46 insertions(+), 6 deletions(-) diff --git a/src/common/Home.js b/src/common/Home.js index 38e32b21..49ed8dde 100644 --- a/src/common/Home.js +++ b/src/common/Home.js @@ -1,7 +1,12 @@ -import React from 'react'; -import { connect } from 'react-redux'; -import './Home.scss'; -import { resetSelectedServer } from '../servers/reducers/selectedServer'; +import chevronIcon from '@fortawesome/fontawesome-free-solid/faChevronRight' +import FontAwesomeIcon from '@fortawesome/react-fontawesome' +import { isEmpty, pick, values } from 'ramda' +import React from 'react' +import { connect } from 'react-redux' +import { Link } from 'react-router-dom' +import { ListGroup, ListGroupItem } from 'reactstrap' +import { resetSelectedServer } from '../servers/reducers/selectedServer' +import './Home.scss' export class Home extends React.Component { componentDidMount() { @@ -9,13 +14,30 @@ export class Home extends React.Component { } render() { + const servers = values(this.props.servers); + const hasServers = !isEmpty(servers); + return (

Welcome to Shlink

-
Please, select a server.
+
+ {hasServers && Please, select a server.} + {!hasServers && Please, add a server.} +
+ + {hasServers && ( + + {servers.map(({ name, id }) => ( + + {name} + + + ))} + + )}
); } } -export default connect(null, { resetSelectedServer })(Home); +export default connect(pick(['servers']), { resetSelectedServer })(Home); diff --git a/src/common/Home.scss b/src/common/Home.scss index 7641664b..8fd117b1 100644 --- a/src/common/Home.scss +++ b/src/common/Home.scss @@ -1,4 +1,5 @@ @import '../utils/base'; +@import '../utils/mixins/vertical-align'; .home-container { text-align: center; @@ -12,3 +13,20 @@ .home-container__title { font-size: 36px; } + +.home-container__servers-list { + margin-top: 1rem; + width: 100%; + max-width: 400px; +} + +.home-container__servers-item.home-container__servers-item { + text-align: left; + position: relative; + padding: .75rem 2.5rem 0.75rem 1rem; +} + +.home-container__servers-item-icon { + @include vertical-align(); + right: 1rem; +} From 49f0109d2019d4b77d5be502680e58d82410253f Mon Sep 17 00:00:00 2001 From: Alejandro Celaya Date: Sun, 12 Aug 2018 08:01:35 +0200 Subject: [PATCH 12/24] Renamed home-container CSS class to just home --- src/common/Home.js | 12 ++++++------ src/common/Home.scss | 10 +++++----- 2 files changed, 11 insertions(+), 11 deletions(-) diff --git a/src/common/Home.js b/src/common/Home.js index 49ed8dde..3455cda4 100644 --- a/src/common/Home.js +++ b/src/common/Home.js @@ -18,19 +18,19 @@ export class Home extends React.Component { const hasServers = !isEmpty(servers); return ( -
-

Welcome to Shlink

-
+
+

Welcome to Shlink

+
{hasServers && Please, select a server.} {!hasServers && Please, add a server.}
{hasServers && ( - + {servers.map(({ name, id }) => ( - + {name} - + ))} diff --git a/src/common/Home.scss b/src/common/Home.scss index 8fd117b1..0333f63c 100644 --- a/src/common/Home.scss +++ b/src/common/Home.scss @@ -1,7 +1,7 @@ @import '../utils/base'; @import '../utils/mixins/vertical-align'; -.home-container { +.home { text-align: center; height: calc(100vh - #{$headerHeight}); display: flex; @@ -10,23 +10,23 @@ flex-flow: column; } -.home-container__title { +.home__title { font-size: 36px; } -.home-container__servers-list { +.home__servers-list { margin-top: 1rem; width: 100%; max-width: 400px; } -.home-container__servers-item.home-container__servers-item { +.home__servers-item.home__servers-item { text-align: left; position: relative; padding: .75rem 2.5rem 0.75rem 1rem; } -.home-container__servers-item-icon { +.home__servers-item-icon { @include vertical-align(); right: 1rem; } From 073703ef5b8495944767c35cb3846655e4309048 Mon Sep 17 00:00:00 2001 From: Alejandro Celaya Date: Sun, 12 Aug 2018 08:20:35 +0200 Subject: [PATCH 13/24] Created Home component tests --- package.json | 3 +- src/common/Home.js | 7 ++++- test/home/Home.test.js | 52 +++++++++++++++++++++++++++++++++ yarn.lock | 66 +++++++++++++++++++++++++++++++++++++++++- 4 files changed, 125 insertions(+), 3 deletions(-) create mode 100644 test/home/Home.test.js diff --git a/package.json b/package.json index bd8f411a..ca67eded 100644 --- a/package.json +++ b/package.json @@ -6,7 +6,7 @@ "scripts": { "start": "node scripts/start.js", "build": "node scripts/build.js", - "test": "node scripts/test.js --env=jsdom" + "test": "node scripts/test.js --env=jsdom --colors" }, "dependencies": { "@fortawesome/fontawesome": "^1.1.8", @@ -70,6 +70,7 @@ "react-dev-utils": "^5.0.1", "resolve": "1.6.0", "sass-loader": "^7.0.1", + "sinon": "^6.1.5", "style-loader": "0.19.0", "sw-precache-webpack-plugin": "0.11.4", "url-loader": "0.6.2", diff --git a/src/common/Home.js b/src/common/Home.js index 3455cda4..8efcd317 100644 --- a/src/common/Home.js +++ b/src/common/Home.js @@ -28,7 +28,12 @@ export class Home extends React.Component { {hasServers && ( {servers.map(({ name, id }) => ( - + {name} diff --git a/test/home/Home.test.js b/test/home/Home.test.js new file mode 100644 index 00000000..52af9f97 --- /dev/null +++ b/test/home/Home.test.js @@ -0,0 +1,52 @@ +import { shallow } from 'enzyme' +import { values } from 'ramda' +import React from 'react' +import * as sinon from 'sinon' +import { Home } from '../../src/common/Home' + +describe('', () => { + let wrapped; + const defaultProps = { + resetSelectedServer: () => {}, + servers: {}, + }; + const createComponent = props => { + const actualProps = { ...defaultProps, ...props }; + wrapped = shallow(); + return wrapped; + }; + + afterEach(() => { + if (wrapped !== undefined) { + wrapped.unmount(); + wrapped = undefined; + } + }); + + it('resets selected server when mounted', () => { + const resetSelectedServer = sinon.spy(); + + expect(resetSelectedServer.called).toEqual(false); + createComponent({ resetSelectedServer }); + expect(resetSelectedServer.called).toEqual(true); + }); + + it('shows link to create server when no servers exist', () => { + const wrapped = createComponent(); + + expect(wrapped.find('Link')).toHaveLength(1); + expect(wrapped.find('ListGroup')).toHaveLength(0); + }); + + it('shows servers list when list of servers is not empty', () => { + const servers = { + 1: { name: 'foo', id: '123' }, + 2: { name: 'bar', id: '456' }, + } + const wrapped = createComponent({ servers }); + + expect(wrapped.find('Link')).toHaveLength(0); + expect(wrapped.find('ListGroup')).toHaveLength(1); + expect(wrapped.find('ListGroupItem')).toHaveLength(values(servers).length); + }); +}); diff --git a/yarn.lock b/yarn.lock index 8e47e890..7c4c492c 100644 --- a/yarn.lock +++ b/yarn.lock @@ -31,6 +31,22 @@ humps "^2.0.1" prop-types "^15.5.7" +"@sinonjs/commons@^1.0.1": + version "1.0.2" + resolved "https://registry.yarnpkg.com/@sinonjs/commons/-/commons-1.0.2.tgz#3e0ac737781627b8844257fadc3d803997d0526e" + dependencies: + type-detect "4.0.8" + +"@sinonjs/formatio@^2.0.0": + version "2.0.0" + resolved "https://registry.yarnpkg.com/@sinonjs/formatio/-/formatio-2.0.0.tgz#84db7e9eb5531df18a8c5e0bfb6e449e55e654b2" + dependencies: + samsam "1.3.0" + +"@sinonjs/samsam@^2.0.0": + version "2.0.0" + resolved "https://registry.yarnpkg.com/@sinonjs/samsam/-/samsam-2.0.0.tgz#9163742ac35c12d3602dece74317643b35db6a80" + "@types/node@*": version "10.5.6" resolved "https://registry.yarnpkg.com/@types/node/-/node-10.5.6.tgz#1640f021dd0eaf12e731e54198c12ad2e020dc8e" @@ -2151,7 +2167,7 @@ detect-port-alt@1.1.6: address "^1.0.1" debug "^2.6.0" -diff@^3.2.0: +diff@^3.2.0, diff@^3.5.0: version "3.5.0" resolved "https://registry.yarnpkg.com/diff/-/diff-3.5.0.tgz#800c0dd1e0a8bfbc95835c202ad220fe317e5a12" @@ -4451,6 +4467,10 @@ jsx-ast-utils@^2.0.0: dependencies: array-includes "^3.0.3" +just-extend@^1.1.27: + version "1.1.27" + resolved "https://registry.yarnpkg.com/just-extend/-/just-extend-1.1.27.tgz#ec6e79410ff914e472652abfa0e603c03d60e905" + killable@^1.0.0: version "1.0.0" resolved "https://registry.yarnpkg.com/killable/-/killable-1.0.0.tgz#da8b84bd47de5395878f95d64d02f2449fe05e6b" @@ -4598,6 +4618,10 @@ lodash.flattendeep@^4.4.0: version "4.4.0" resolved "https://registry.yarnpkg.com/lodash.flattendeep/-/lodash.flattendeep-4.4.0.tgz#fb030917f86a3134e5bc9bec0d69e0013ddfedb2" +lodash.get@^4.4.2: + version "4.4.2" + resolved "https://registry.yarnpkg.com/lodash.get/-/lodash.get-4.4.2.tgz#2d177f652fa31e939b4438d5341499dfa3825e99" + lodash.isfunction@^3.0.9: version "3.0.9" resolved "https://registry.yarnpkg.com/lodash.isfunction/-/lodash.isfunction-3.0.9.tgz#06de25df4db327ac931981d1bdb067e5af68d051" @@ -4647,6 +4671,10 @@ loglevel@^1.4.1: version "1.6.1" resolved "https://registry.yarnpkg.com/loglevel/-/loglevel-1.6.1.tgz#e0fc95133b6ef276cdc8887cdaf24aa6f156f8fa" +lolex@^2.3.2, lolex@^2.7.1: + version "2.7.1" + resolved "https://registry.yarnpkg.com/lolex/-/lolex-2.7.1.tgz#e40a8c4d1f14b536aa03e42a537c7adbaf0c20be" + longest@^1.0.1: version "1.0.1" resolved "https://registry.yarnpkg.com/longest/-/longest-1.0.1.tgz#30a0b2da38f73770e8294a0d22e6625ed77d0097" @@ -4975,6 +5003,16 @@ next-tick@1: version "1.0.0" resolved "https://registry.yarnpkg.com/next-tick/-/next-tick-1.0.0.tgz#ca86d1fe8828169b0120208e3dc8424b9db8342c" +nise@^1.4.2: + version "1.4.2" + resolved "https://registry.yarnpkg.com/nise/-/nise-1.4.2.tgz#a9a3800e3994994af9e452333d549d60f72b8e8c" + dependencies: + "@sinonjs/formatio" "^2.0.0" + just-extend "^1.1.27" + lolex "^2.3.2" + path-to-regexp "^1.7.0" + text-encoding "^0.6.4" + no-case@^2.2.0: version "2.3.2" resolved "https://registry.yarnpkg.com/no-case/-/no-case-2.3.2.tgz#60b813396be39b3f1288a4c1ed5d1e7d28b464ac" @@ -6623,6 +6661,10 @@ safe-regex@^1.1.0: version "2.1.2" resolved "https://registry.yarnpkg.com/safer-buffer/-/safer-buffer-2.1.2.tgz#44fa161b0187b9549dd84bb91802f9bd8385cd6a" +samsam@1.3.0: + version "1.3.0" + resolved "https://registry.yarnpkg.com/samsam/-/samsam-1.3.0.tgz#8d1d9350e25622da30de3e44ba692b5221ab7c50" + sane@~1.6.0: version "1.6.0" resolved "https://registry.yarnpkg.com/sane/-/sane-1.6.0.tgz#9610c452307a135d29c1fdfe2547034180c46775" @@ -6819,6 +6861,20 @@ signal-exit@^3.0.0, signal-exit@^3.0.2: version "3.0.2" resolved "https://registry.yarnpkg.com/signal-exit/-/signal-exit-3.0.2.tgz#b5fdc08f1287ea1178628e415e25132b73646c6d" +sinon@^6.1.5: + version "6.1.5" + resolved "https://registry.yarnpkg.com/sinon/-/sinon-6.1.5.tgz#41451502d43cd5ffb9d051fbf507952400e81d09" + dependencies: + "@sinonjs/commons" "^1.0.1" + "@sinonjs/formatio" "^2.0.0" + "@sinonjs/samsam" "^2.0.0" + diff "^3.5.0" + lodash.get "^4.4.2" + lolex "^2.7.1" + nise "^1.4.2" + supports-color "^5.4.0" + type-detect "^4.0.8" + slash@^1.0.0: version "1.0.0" resolved "https://registry.yarnpkg.com/slash/-/slash-1.0.0.tgz#c41f2f6c39fc16d1cd17ad4b5d896114ae470d55" @@ -7237,6 +7293,10 @@ test-exclude@^4.2.1: read-pkg-up "^1.0.1" require-main-filename "^1.0.1" +text-encoding@^0.6.4: + version "0.6.4" + resolved "https://registry.yarnpkg.com/text-encoding/-/text-encoding-0.6.4.tgz#e399a982257a276dae428bb92845cb71bdc26d19" + text-table@0.2.0, text-table@~0.2.0: version "0.2.0" resolved "https://registry.yarnpkg.com/text-table/-/text-table-0.2.0.tgz#7f5ee823ae805207c00af2df4a84ec3fcfa570b4" @@ -7366,6 +7426,10 @@ type-check@~0.3.2: dependencies: prelude-ls "~1.1.2" +type-detect@4.0.8, type-detect@^4.0.8: + version "4.0.8" + resolved "https://registry.yarnpkg.com/type-detect/-/type-detect-4.0.8.tgz#7646fb5f18871cfbb7749e69bd39a6388eb7450c" + type-is@~1.6.15, type-is@~1.6.16: version "1.6.16" resolved "https://registry.yarnpkg.com/type-is/-/type-is-1.6.16.tgz#f89ce341541c672b25ee7ae3c73dee3b2be50194" From 86eb963176f9fc523881bda5c37893120c9746de Mon Sep 17 00:00:00 2001 From: Alejandro Celaya Date: Sun, 12 Aug 2018 08:26:36 +0200 Subject: [PATCH 14/24] Simplified AsideMenu component removing unneeded checks --- src/common/AsideMenu.js | 20 +++++++------------- 1 file changed, 7 insertions(+), 13 deletions(-) diff --git a/src/common/AsideMenu.js b/src/common/AsideMenu.js index a098d4cf..e9fa832f 100644 --- a/src/common/AsideMenu.js +++ b/src/common/AsideMenu.js @@ -1,18 +1,13 @@ -import listIcon from '@fortawesome/fontawesome-free-solid/faBars'; -import createIcon from '@fortawesome/fontawesome-free-solid/faPlus'; -import FontAwesomeIcon from '@fortawesome/react-fontawesome'; -import React from 'react'; -import { NavLink } from 'react-router-dom'; -import DeleteServerButton from '../servers/DeleteServerButton'; -import './AsideMenu.scss'; +import listIcon from '@fortawesome/fontawesome-free-solid/faBars' +import createIcon from '@fortawesome/fontawesome-free-solid/faPlus' +import FontAwesomeIcon from '@fortawesome/react-fontawesome' +import React from 'react' +import { NavLink } from 'react-router-dom' +import DeleteServerButton from '../servers/DeleteServerButton' +import './AsideMenu.scss' export default function AsideMenu({ selectedServer, history }) { const serverId = selectedServer ? selectedServer.id : ''; - const isListShortUrlsActive = (match, { pathname }) => { - // FIXME. Should use the 'match' params, but they are not being properly resolved. Investigate - const serverIdFromPathname = pathname.split('/')[2]; - return serverIdFromPathname === serverId && pathname.indexOf('list-short-urls') !== -1; - }; return ( ); } + +AsideMenu.propTypes = { + selectedServer: PropTypes.shape({ + id: PropTypes.string, + name: PropTypes.string, + url: PropTypes.string, + apiKey: PropTypes.string, + }), +}; diff --git a/src/common/MenuLayout.js b/src/common/MenuLayout.js index 7105746a..4b3a983f 100644 --- a/src/common/MenuLayout.js +++ b/src/common/MenuLayout.js @@ -16,9 +16,11 @@ export class MenuLayout extends React.Component { } render() { + const { selectedServer } = this.props; + return (
- +
this.setState({ isModalOpen: !this.state.isModalOpen })} - history={history} server={server} key="deleteServerModal" /> diff --git a/src/servers/DeleteServerModal.js b/src/servers/DeleteServerModal.js index b0f62fad..95a0b554 100644 --- a/src/servers/DeleteServerModal.js +++ b/src/servers/DeleteServerModal.js @@ -1,9 +1,12 @@ +import PropTypes from 'prop-types'; import React from 'react'; import { connect } from 'react-redux'; +import { withRouter } from 'react-router-dom'; import { Modal, ModalBody, ModalFooter, ModalHeader } from 'reactstrap'; +import { compose } from 'redux'; import { deleteServer } from './reducers/server'; -export const DeleteServerModal = ({ server, deleteServer, toggle, history, isOpen }) => { +export const DeleteServerModal = ({ server, toggle, isOpen, deleteServer, history }) => { const closeModal = () => { deleteServer(server); toggle(); @@ -15,7 +18,10 @@ export const DeleteServerModal = ({ server, deleteServer, toggle, history, isOpe Delete server

Are you sure you want to delete server {server ? server.name : ''}?

-

No data will be deleted, only the access to that server will be removed from this host. You can create it again at any moment.

+

+ No data will be deleted, only the access to that server will be removed from this host. + You can create it again at any moment. +

@@ -25,4 +31,18 @@ export const DeleteServerModal = ({ server, deleteServer, toggle, history, isOpe ); }; -export default connect(null, { deleteServer })(DeleteServerModal); +DeleteServerModal.propTypes = { + toggle: PropTypes.func.isRequired, + isOpen: PropTypes.bool.isRequired, + server: PropTypes.shape({ + id: PropTypes.string, + name: PropTypes.string, + url: PropTypes.string, + apiKey: PropTypes.string, + }), +}; + +export default compose( + withRouter, + connect(null, { deleteServer }) +)(DeleteServerModal); diff --git a/test/home/Home.test.js b/test/common/Home.test.js similarity index 88% rename from test/home/Home.test.js rename to test/common/Home.test.js index 52af9f97..1495604f 100644 --- a/test/home/Home.test.js +++ b/test/common/Home.test.js @@ -1,8 +1,8 @@ -import { shallow } from 'enzyme' -import { values } from 'ramda' -import React from 'react' -import * as sinon from 'sinon' -import { Home } from '../../src/common/Home' +import { shallow } from 'enzyme'; +import { values } from 'ramda'; +import React from 'react'; +import * as sinon from 'sinon'; +import { Home } from '../../src/common/Home'; describe('', () => { let wrapped; From faa828c58ab257c7182d1a36962cb4da1e5cb510 Mon Sep 17 00:00:00 2001 From: Alejandro Celaya Date: Sun, 12 Aug 2018 08:49:08 +0200 Subject: [PATCH 16/24] Created AsideMenu component test --- test/common/AsideMenu.test.js | 25 +++++++++++++++++++++++++ 1 file changed, 25 insertions(+) create mode 100644 test/common/AsideMenu.test.js diff --git a/test/common/AsideMenu.test.js b/test/common/AsideMenu.test.js new file mode 100644 index 00000000..db5e8f71 --- /dev/null +++ b/test/common/AsideMenu.test.js @@ -0,0 +1,25 @@ +import { shallow } from 'enzyme' +import React from 'react' +import AsideMenu from '../../src/common/AsideMenu' + +describe('', () => { + let wrapped; + + beforeEach(() => { + wrapped = shallow(); + }); + afterEach(() => { + wrapped.unmount(); + }); + + it('contains links to selected server', () => { + const links = wrapped.find('NavLink'); + + expect(links).toHaveLength(2); + links.forEach(link => expect(link.prop('to')).toContain('abc123')); + }); + + it('contains a button to delete server', () => { + expect(wrapped.find('DeleteServerButton')).toHaveLength(1); + }); +}); From f23245a39cbcfc718d437cf28e5109ea7c3cee0e Mon Sep 17 00:00:00 2001 From: Alejandro Celaya Date: Sun, 12 Aug 2018 09:01:11 +0200 Subject: [PATCH 17/24] Created DateInput component test --- src/common/DateInput.js | 12 ++++++----- test/common/DateInput.test.js | 38 +++++++++++++++++++++++++++++++++++ 2 files changed, 45 insertions(+), 5 deletions(-) create mode 100644 test/common/DateInput.test.js diff --git a/src/common/DateInput.js b/src/common/DateInput.js index b246405c..2981810a 100644 --- a/src/common/DateInput.js +++ b/src/common/DateInput.js @@ -24,11 +24,13 @@ export default class DateInput extends React.Component { readOnly ref={this.inputRef} /> - {showCalendarIcon && this.inputRef.current.input.focus()} - />} + {showCalendarIcon && ( + this.inputRef.current.input.focus()} + /> + )}
); } diff --git a/test/common/DateInput.test.js b/test/common/DateInput.test.js new file mode 100644 index 00000000..fe68ece9 --- /dev/null +++ b/test/common/DateInput.test.js @@ -0,0 +1,38 @@ +import React from 'react'; +import { shallow } from 'enzyme'; +import DateInput from '../../src/common/DateInput'; +import FontAwesomeIcon from '@fortawesome/react-fontawesome'; + +describe('', () => { + let wrapped; + + const createComponent = (props = {}) => { + wrapped = shallow(); + return wrapped; + }; + afterEach(() => { + if (wrapped !== undefined) { + wrapped.unmount(); + wrapped = undefined; + } + }); + + it('wrapps a DatePicker', () => { + wrapped = createComponent(); + }); + + it('shows calendar icon when input is not clearable', () => { + wrapped = createComponent({ isClearable: false }); + expect(wrapped.find(FontAwesomeIcon)).toHaveLength(1); + }); + + it('shows calendar icon when input is clearable but selected value is nil', () => { + wrapped = createComponent({ isClearable: true, selected: null }); + expect(wrapped.find(FontAwesomeIcon)).toHaveLength(1); + }); + + it('does not show calendar icon when input is clearable', () => { + wrapped = createComponent({ isClearable: true, selected: '' }); + expect(wrapped.find(FontAwesomeIcon)).toHaveLength(0); + }); +}); From ec4c14e8dedcf96bbe66a28bd004b82497a064e8 Mon Sep 17 00:00:00 2001 From: Alejandro Celaya Date: Sun, 12 Aug 2018 09:22:18 +0200 Subject: [PATCH 18/24] Created selectedServer reducer test --- src/servers/reducers/selectedServer.js | 8 +-- .../reducers/shortUrlsListParams.js | 2 +- test/servers/reducers/selectedServer.test.js | 54 +++++++++++++++++++ 3 files changed, 60 insertions(+), 4 deletions(-) create mode 100644 test/servers/reducers/selectedServer.test.js diff --git a/src/servers/reducers/selectedServer.js b/src/servers/reducers/selectedServer.js index 18833b34..3ed83fa6 100644 --- a/src/servers/reducers/selectedServer.js +++ b/src/servers/reducers/selectedServer.js @@ -1,9 +1,10 @@ import ShlinkApiClient from '../../api/ShlinkApiClient'; import ServersService from '../../servers/services/ServersService'; import { resetShortUrlParams } from '../../short-urls/reducers/shortUrlsListParams' +import { curry } from 'ramda'; -const SELECT_SERVER = 'shlink/selectedServer/SELECT_SERVER'; -const RESET_SELECTED_SERVER = 'shlink/selectedServer/RESET_SELECTED_SERVER'; +export const SELECT_SERVER = 'shlink/selectedServer/SELECT_SERVER'; +export const RESET_SELECTED_SERVER = 'shlink/selectedServer/RESET_SELECTED_SERVER'; const defaultState = null; @@ -20,7 +21,7 @@ export default function reducer(state = defaultState, action) { export const resetSelectedServer = () => ({ type: RESET_SELECTED_SERVER }); -export const selectServer = serverId => dispatch => { +export const _selectServer = (ShlinkApiClient, ServersService, serverId) => dispatch => { dispatch(resetShortUrlParams()); const selectedServer = ServersService.findServerById(serverId); @@ -31,3 +32,4 @@ export const selectServer = serverId => dispatch => { selectedServer }) }; +export const selectServer = curry(_selectServer)(ShlinkApiClient, ServersService); diff --git a/src/short-urls/reducers/shortUrlsListParams.js b/src/short-urls/reducers/shortUrlsListParams.js index 3217167d..29464264 100644 --- a/src/short-urls/reducers/shortUrlsListParams.js +++ b/src/short-urls/reducers/shortUrlsListParams.js @@ -1,6 +1,6 @@ import { LIST_SHORT_URLS } from './shortUrlsList'; -const RESET_SHORT_URL_PARAMS = 'shlink/shortUrlsListParams/RESET_SHORT_URL_PARAMS'; +export const RESET_SHORT_URL_PARAMS = 'shlink/shortUrlsListParams/RESET_SHORT_URL_PARAMS'; const defaultState = { page: '1' }; diff --git a/test/servers/reducers/selectedServer.test.js b/test/servers/reducers/selectedServer.test.js new file mode 100644 index 00000000..731655ab --- /dev/null +++ b/test/servers/reducers/selectedServer.test.js @@ -0,0 +1,54 @@ +import { + _selectServer, + RESET_SELECTED_SERVER, + resetSelectedServer, + SELECT_SERVER, +} from '../../../src/servers/reducers/selectedServer'; +import * as sinon from 'sinon'; +import { RESET_SHORT_URL_PARAMS } from '../../../src/short-urls/reducers/shortUrlsListParams'; + +describe('selectedServerReducer', () => { + describe('resetSelectedServer', () => { + it('returns proper action', () => { + expect(resetSelectedServer()).toEqual({ type: RESET_SELECTED_SERVER }); + }); + }); + + describe('selectedServer', () => { + const ShlinkApiClientMock = { + setConfig: sinon.spy() + }; + const serverId = 'abc123'; + const selectedServer = { + id: serverId + }; + const ServersServiceMock = { + findServerById: sinon.fake.returns(selectedServer) + }; + + afterEach(() => { + ShlinkApiClientMock.setConfig.resetHistory(); + ServersServiceMock.findServerById.resetHistory(); + }); + + it('dispatches proper actions', () => { + const dispatch = sinon.spy(); + + _selectServer(ShlinkApiClientMock, ServersServiceMock, serverId)(dispatch); + + expect(dispatch.callCount).toEqual(2); + expect(dispatch.firstCall.calledWith({ type: RESET_SHORT_URL_PARAMS })).toEqual(true); + expect(dispatch.secondCall.calledWith({ + type: SELECT_SERVER, + selectedServer + })).toEqual(true); + }); + + it('invokes dependencies', () => { + _selectServer(ShlinkApiClientMock, ServersServiceMock, serverId)(() => {}); + + expect(ShlinkApiClientMock.setConfig.callCount).toEqual(1); + expect(ServersServiceMock.findServerById.callCount).toEqual(1); + }); + }); +}); From 6969233b6ff85e2db5304698a5a04a9805752793 Mon Sep 17 00:00:00 2001 From: Alejandro Celaya Date: Sun, 12 Aug 2018 09:34:14 +0200 Subject: [PATCH 19/24] Added reducer test to selectedServerReducer test --- test/servers/reducers/selectedServer.test.js | 19 +++++++++++++++++-- 1 file changed, 17 insertions(+), 2 deletions(-) diff --git a/test/servers/reducers/selectedServer.test.js b/test/servers/reducers/selectedServer.test.js index 731655ab..ca73ede4 100644 --- a/test/servers/reducers/selectedServer.test.js +++ b/test/servers/reducers/selectedServer.test.js @@ -1,4 +1,4 @@ -import { +import reduce, { _selectServer, RESET_SELECTED_SERVER, resetSelectedServer, @@ -8,13 +8,28 @@ import * as sinon from 'sinon'; import { RESET_SHORT_URL_PARAMS } from '../../../src/short-urls/reducers/shortUrlsListParams'; describe('selectedServerReducer', () => { + describe('reduce', () => { + it('returns default when action is not handled', () => + expect(reduce(null, { type: 'unknown' })).toEqual(null) + ); + + it('returns default when action is RESET_SELECTED_SERVER', () => + expect(reduce(null, { type: RESET_SELECTED_SERVER })).toEqual(null) + ); + + it('returns selected server when action is SELECT_SERVER', () => { + const selectedServer = { id: 'abc123' }; + expect(reduce(null, { type: SELECT_SERVER, selectedServer })).toEqual(selectedServer); + }); + }); + describe('resetSelectedServer', () => { it('returns proper action', () => { expect(resetSelectedServer()).toEqual({ type: RESET_SELECTED_SERVER }); }); }); - describe('selectedServer', () => { + describe('selectServer', () => { const ShlinkApiClientMock = { setConfig: sinon.spy() }; From e0ab67899d2085f071bdc1208f95a29f9f108da4 Mon Sep 17 00:00:00 2001 From: Alejandro Celaya Date: Sun, 12 Aug 2018 10:17:13 +0200 Subject: [PATCH 20/24] Created server reducer test --- src/servers/reducers/server.js | 28 ++++----- test/servers/reducers/server.test.js | 87 ++++++++++++++++++++++++++++ 2 files changed, 102 insertions(+), 13 deletions(-) create mode 100644 test/servers/reducers/server.test.js diff --git a/src/servers/reducers/server.js b/src/servers/reducers/server.js index 5daa566b..08c86c2e 100644 --- a/src/servers/reducers/server.js +++ b/src/servers/reducers/server.js @@ -1,8 +1,9 @@ import ServersService from '../services/ServersService'; +import { curry } from 'ramda'; -const FETCH_SERVERS = 'shlink/servers/FETCH_SERVERS'; -const CREATE_SERVER = 'shlink/servers/CREATE_SERVER'; -const DELETE_SERVER = 'shlink/servers/DELETE_SERVER'; +export const FETCH_SERVERS = 'shlink/servers/FETCH_SERVERS'; +export const CREATE_SERVER = 'shlink/servers/CREATE_SERVER'; +export const DELETE_SERVER = 'shlink/servers/DELETE_SERVER'; export default function reducer(state = {}, action) { switch (action.type) { @@ -17,19 +18,20 @@ export default function reducer(state = {}, action) { } } -export const listServers = () => { - return { - type: FETCH_SERVERS, - servers: ServersService.listServers(), - }; -}; +export const _listServers = ServersService => ({ + type: FETCH_SERVERS, + servers: ServersService.listServers(), +}); +export const listServers = () => _listServers(ServersService); -export const createServer = server => { +export const _createServer = (ServersService, server) => { ServersService.createServer(server); - return listServers(); + return _listServers(ServersService); }; +export const createServer = curry(_createServer)(ServersService); -export const deleteServer = server => { +export const _deleteServer = (ServersService, server) => { ServersService.deleteServer(server); - return listServers(); + return _listServers(ServersService); }; +export const deleteServer = curry(_deleteServer)(ServersService); diff --git a/test/servers/reducers/server.test.js b/test/servers/reducers/server.test.js new file mode 100644 index 00000000..cd3f24b1 --- /dev/null +++ b/test/servers/reducers/server.test.js @@ -0,0 +1,87 @@ +import reduce, { + _createServer, + _deleteServer, + _listServers, + CREATE_SERVER, + DELETE_SERVER, + FETCH_SERVERS, +} from '../../../src/servers/reducers/server'; +import * as sinon from 'sinon'; + +describe('serverReducer', () => { + const servers = { + abc123: { id: 'abc123' }, + def456: { id: 'def456' } + }; + const ServersServiceMock = { + listServers: sinon.fake.returns(servers), + createServer: sinon.fake(), + deleteServer: sinon.fake(), + }; + + describe('reduce', () => { + it('returns servers when action is FETCH_SERVERS', () => + expect(reduce({}, { type: FETCH_SERVERS, servers })).toEqual(servers) + ); + + it('returns servers when action is DELETE_SERVER', () => + expect(reduce({}, { type: DELETE_SERVER, servers })).toEqual(servers) + ); + + it('adds server to list when action is CREATE_SERVER', () => { + const server = { id: 'abc123' }; + expect(reduce({}, { type: CREATE_SERVER, server })).toEqual({ + [server.id]: server + }) + }); + + it('returns default when action is unknown', () => + expect(reduce({}, { type: 'unknown' })).toEqual({}) + ); + }); + + describe('action creators', () => { + beforeEach(() => { + ServersServiceMock.listServers.resetHistory(); + ServersServiceMock.createServer.resetHistory(); + ServersServiceMock.deleteServer.resetHistory(); + }); + + describe('listServers', () => { + it('fetches servers and returns them as part of the action', () => { + const result = _listServers(ServersServiceMock); + + expect(result).toEqual({ type: FETCH_SERVERS, servers }); + expect(ServersServiceMock.listServers.callCount).toEqual(1); + expect(ServersServiceMock.createServer.callCount).toEqual(0); + expect(ServersServiceMock.deleteServer.callCount).toEqual(0); + }); + }); + + describe('createServer', () => { + it('adds new server and then fetches servers again', () => { + const serverToCreate = { id: 'abc123' }; + const result = _createServer(ServersServiceMock, serverToCreate); + + expect(result).toEqual({ type: FETCH_SERVERS, servers }); + expect(ServersServiceMock.listServers.callCount).toEqual(1); + expect(ServersServiceMock.createServer.callCount).toEqual(1); + expect(ServersServiceMock.createServer.firstCall.calledWith(serverToCreate)).toEqual(true); + expect(ServersServiceMock.deleteServer.callCount).toEqual(0); + }); + }); + + describe('deleteServer', () => { + it('deletes a server and then fetches servers again', () => { + const serverToDelete = { id: 'abc123' }; + const result = _deleteServer(ServersServiceMock, serverToDelete); + + expect(result).toEqual({ type: FETCH_SERVERS, servers }); + expect(ServersServiceMock.listServers.callCount).toEqual(1); + expect(ServersServiceMock.createServer.callCount).toEqual(0); + expect(ServersServiceMock.deleteServer.callCount).toEqual(1); + expect(ServersServiceMock.deleteServer.firstCall.calledWith(serverToDelete)).toEqual(true); + }); + }); + }); +}); From d6e6c8c6c2e42f50228c483ca144227e54a43437 Mon Sep 17 00:00:00 2001 From: Alejandro Celaya Date: Sun, 12 Aug 2018 10:18:26 +0200 Subject: [PATCH 21/24] Fixed wrong value passed to DateInput --- test/common/DateInput.test.js | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/test/common/DateInput.test.js b/test/common/DateInput.test.js index fe68ece9..a2df216e 100644 --- a/test/common/DateInput.test.js +++ b/test/common/DateInput.test.js @@ -2,6 +2,7 @@ import React from 'react'; import { shallow } from 'enzyme'; import DateInput from '../../src/common/DateInput'; import FontAwesomeIcon from '@fortawesome/react-fontawesome'; +import moment from 'moment'; describe('', () => { let wrapped; @@ -32,7 +33,7 @@ describe('', () => { }); it('does not show calendar icon when input is clearable', () => { - wrapped = createComponent({ isClearable: true, selected: '' }); + wrapped = createComponent({ isClearable: true, selected: moment() }); expect(wrapped.find(FontAwesomeIcon)).toHaveLength(0); }); }); From f8eb5fb022bce0dcfa839b1252c1f23c9e030c39 Mon Sep 17 00:00:00 2001 From: Alejandro Celaya Date: Sun, 12 Aug 2018 18:49:57 +0200 Subject: [PATCH 22/24] Creates shortUrlsListParams reducer test --- .../reducers/shortUrlCreationResult.js | 4 ++- src/short-urls/reducers/shortUrlVisits.js | 4 ++- src/short-urls/reducers/shortUrlsList.js | 4 ++- .../reducers/shortUrlsListParams.test.js | 32 +++++++++++++++++++ 4 files changed, 41 insertions(+), 3 deletions(-) create mode 100644 test/shortUrls/reducers/shortUrlsListParams.test.js diff --git a/src/short-urls/reducers/shortUrlCreationResult.js b/src/short-urls/reducers/shortUrlCreationResult.js index 88675f6c..c74b97d5 100644 --- a/src/short-urls/reducers/shortUrlCreationResult.js +++ b/src/short-urls/reducers/shortUrlCreationResult.js @@ -1,4 +1,5 @@ import ShlinkApiClient from '../../api/ShlinkApiClient'; +import { curry } from 'ramda'; const CREATE_SHORT_URL_START = 'shlink/createShortUrl/CREATE_SHORT_URL_START'; const CREATE_SHORT_URL_ERROR = 'shlink/createShortUrl/CREATE_SHORT_URL_ERROR'; @@ -37,7 +38,7 @@ export default function reducer(state = defaultState, action) { } } -export const createShortUrl = data => async dispatch => { +export const _createShortUrl = (ShlinkApiClient, data) => async dispatch => { dispatch({ type: CREATE_SHORT_URL_START }); try { @@ -47,5 +48,6 @@ export const createShortUrl = data => async dispatch => { dispatch({ type: CREATE_SHORT_URL_ERROR }); } }; +export const createShortUrl = curry(_createShortUrl)(ShlinkApiClient); export const resetCreateShortUrl = () => ({ type: RESET_CREATE_SHORT_URL }); diff --git a/src/short-urls/reducers/shortUrlVisits.js b/src/short-urls/reducers/shortUrlVisits.js index 0f9c8ed4..a247fa2a 100644 --- a/src/short-urls/reducers/shortUrlVisits.js +++ b/src/short-urls/reducers/shortUrlVisits.js @@ -1,4 +1,5 @@ import ShlinkApiClient from '../../api/ShlinkApiClient'; +import { curry } from 'ramda'; const GET_SHORT_URL_VISITS_START = 'shlink/shortUrlVisits/GET_SHORT_URL_VISITS_START'; const GET_SHORT_URL_VISITS_ERROR = 'shlink/shortUrlVisits/GET_SHORT_URL_VISITS_ERROR'; @@ -36,7 +37,7 @@ export default function dispatch (state = initialState, action) { } } -export const getShortUrlVisits = (shortCode, dates) => dispatch => { +export const _getShortUrlVisits = (ShlinkApiClient, shortCode, dates) => dispatch => { dispatch({ type: GET_SHORT_URL_VISITS_START }); Promise.all([ @@ -46,3 +47,4 @@ export const getShortUrlVisits = (shortCode, dates) => dispatch => { .then(([visits, shortUrl]) => dispatch({ visits, shortUrl, type: GET_SHORT_URL_VISITS })) .catch(() => dispatch({ type: GET_SHORT_URL_VISITS_ERROR })); }; +export const getShortUrlVisits = curry(_getShortUrlVisits)(ShlinkApiClient); diff --git a/src/short-urls/reducers/shortUrlsList.js b/src/short-urls/reducers/shortUrlsList.js index f9771eae..728d53b0 100644 --- a/src/short-urls/reducers/shortUrlsList.js +++ b/src/short-urls/reducers/shortUrlsList.js @@ -1,4 +1,5 @@ import ShlinkApiClient from '../../api/ShlinkApiClient'; +import { curry } from 'ramda'; const LIST_SHORT_URLS_START = 'shlink/shortUrlsList/LIST_SHORT_URLS_START'; const LIST_SHORT_URLS_ERROR = 'shlink/shortUrlsList/LIST_SHORT_URLS_ERROR'; @@ -30,7 +31,7 @@ export default function reducer(state = initialState, action) { } } -export const listShortUrls = (params = {}) => async dispatch => { +export const _listShortUrls = (ShlinkApiClient, params = {}) => async dispatch => { dispatch({ type: LIST_SHORT_URLS_START }); try { @@ -40,3 +41,4 @@ export const listShortUrls = (params = {}) => async dispatch => { dispatch({ type: LIST_SHORT_URLS_ERROR, params }); } }; +export const listShortUrls = curry(_listShortUrls)(ShlinkApiClient); diff --git a/test/shortUrls/reducers/shortUrlsListParams.test.js b/test/shortUrls/reducers/shortUrlsListParams.test.js new file mode 100644 index 00000000..f22507d3 --- /dev/null +++ b/test/shortUrls/reducers/shortUrlsListParams.test.js @@ -0,0 +1,32 @@ +import reduce, { + RESET_SHORT_URL_PARAMS, + resetShortUrlParams, +} from '../../../src/short-urls/reducers/shortUrlsListParams'; +import { LIST_SHORT_URLS } from '../../../src/short-urls/reducers/shortUrlsList'; + +describe('shortUrlsListParamsReducer', () => { + describe('reduce', () => { + const defaultState = { page: '1' }; + + it('returns default value when action is anknown', () => + expect(reduce(defaultState, { type: 'unknown' })).toEqual(defaultState) + ); + + it('returns params when action is LIST_SHORT_URLS', () => + expect(reduce(defaultState, { type: LIST_SHORT_URLS, params: { searchTerm: 'foo' } })).toEqual({ + ...defaultState, + searchTerm: 'foo' + }) + ); + + it('returns default value when action is RESET_SHORT_URL_PARAMS', () => + expect(reduce(defaultState, { type: RESET_SHORT_URL_PARAMS })).toEqual(defaultState) + ); + }); + + describe('resetShortUrlParams', () => { + it('returns proper action', () => + expect(resetShortUrlParams()).toEqual({ type: RESET_SHORT_URL_PARAMS }) + ); + }); +}); From adec759579bbf5b3a43e8fe4e0b9fbe6656a94a4 Mon Sep 17 00:00:00 2001 From: Alejandro Celaya Date: Sun, 12 Aug 2018 19:07:42 +0200 Subject: [PATCH 23/24] Added workaround to add tags on blur on tags input which allows tags to be added on Android --- src/short-urls/CreateShortUrl.js | 3 ++- src/short-urls/reducers/shortUrlsList.js | 3 +-- 2 files changed, 3 insertions(+), 3 deletions(-) diff --git a/src/short-urls/CreateShortUrl.js b/src/short-urls/CreateShortUrl.js index 971a34f9..59655469 100644 --- a/src/short-urls/CreateShortUrl.js +++ b/src/short-urls/CreateShortUrl.js @@ -71,8 +71,9 @@ export class CreateShortUrl extends React.Component {
diff --git a/src/short-urls/reducers/shortUrlsList.js b/src/short-urls/reducers/shortUrlsList.js index 728d53b0..9d76e44c 100644 --- a/src/short-urls/reducers/shortUrlsList.js +++ b/src/short-urls/reducers/shortUrlsList.js @@ -1,5 +1,4 @@ import ShlinkApiClient from '../../api/ShlinkApiClient'; -import { curry } from 'ramda'; const LIST_SHORT_URLS_START = 'shlink/shortUrlsList/LIST_SHORT_URLS_START'; const LIST_SHORT_URLS_ERROR = 'shlink/shortUrlsList/LIST_SHORT_URLS_ERROR'; @@ -41,4 +40,4 @@ export const _listShortUrls = (ShlinkApiClient, params = {}) => async dispatch = dispatch({ type: LIST_SHORT_URLS_ERROR, params }); } }; -export const listShortUrls = curry(_listShortUrls)(ShlinkApiClient); +export const listShortUrls = (params = {}) => _listShortUrls(ShlinkApiClient, params); From 7f317390e367a6f4be31a5b04550e9447f72d14e Mon Sep 17 00:00:00 2001 From: Alejandro Celaya Date: Sun, 12 Aug 2018 19:18:34 +0200 Subject: [PATCH 24/24] Added v0.2.0 to changelog --- CHANGELOG.md | 26 ++++++++++++++++++++++++++ 1 file changed, 26 insertions(+) diff --git a/CHANGELOG.md b/CHANGELOG.md index 49db70ab..ed8c3c2b 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -1,5 +1,31 @@ # CHANGELOG +## 0.2.0 - 2018-08-12 + +#### Added + +* [#12](https://github.com/shlinkio/shlink-web-client/issues/12) Improved code coverage +* [#20](https://github.com/shlinkio/shlink-web-client/issues/20) Added servers list in welcome page, as well as added link to create one when none exist. + +#### Changed + +* [#11](https://github.com/shlinkio/shlink-web-client/issues/11) Improved app icons fro progressive web apps. + +#### Deprecated + +* *Nothing* + +#### Removed + +* *Nothing* + +#### Fixed + +* [#19](https://github.com/shlinkio/shlink-web-client/issues/19) Added workaround in tags input so that it is possible to add tags on Android devices. +* [#17](https://github.com/shlinkio/shlink-web-client/issues/17) Fixed short URLs list not being sortable in mobile resolutions. +* [#13](https://github.com/shlinkio/shlink-web-client/issues/13) Improved visits page on mobile resolutions. + + ## 0.1.1 - 2018-08-06 #### Added