feat search: search files, depth-first
feat files.view: Open files using Web Activities feat copy/paste: Copy and Paste/Move files fix filters: add "all" filter which clears filters out
This commit is contained in:
		
							
								
								
									
										11
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										11
									
								
								README.md
									
									
									
									
									
								
							| @@ -18,14 +18,15 @@ Please read the Features section below and issues to make sure your issue is not | ||||
| - [x] File Size | ||||
| - [x] Directory Child Count | ||||
| - [x] Actions on multiple files (selection) | ||||
| - [ ] Copy/Cut and Paste files | ||||
| - [ ] File Preview | ||||
| - [ ] Filter Files | ||||
| - [ ] Search | ||||
| - [x] Copy and Paste/Move files | ||||
| - [x] File Preview | ||||
| - [x] Filter Files | ||||
| - [x] Swipe Gestures (Up directory by swiping right) | ||||
| - [x] Search | ||||
| - [ ] Intro | ||||
| - [ ] Different views (List, Icons, etc) | ||||
| - [ ] Share Files | ||||
| - [ ] Preferences | ||||
| - [ ] FTP Browser | ||||
| - [ ] File Type Icons | ||||
| - [ ] Swipe Gestures | ||||
| - [ ] Wi-Fi File Transfer (is this possible?) | ||||
|   | ||||
							
								
								
									
										
											BIN
										
									
								
								build/img/Search.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								build/img/Search.png
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 690 B | 
							
								
								
									
										14
									
								
								build/img/Search.svg
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										14
									
								
								build/img/Search.svg
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,14 @@ | ||||
| <?xml version="1.0" encoding="UTF-8" standalone="no"?> | ||||
| <svg width="19px" height="27px" viewBox="0 0 19 27" version="1.1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" xmlns:sketch="http://www.bohemiancoding.com/sketch/ns"> | ||||
|     <!-- Generator: Sketch 3.3.3 (12072) - http://www.bohemiancoding.com/sketch --> | ||||
|     <title>Search</title> | ||||
|     <desc>Created with Sketch.</desc> | ||||
|     <defs></defs> | ||||
|     <g id="Components" stroke="none" stroke-width="1" fill="none" fill-rule="evenodd" sketch:type="MSPage"> | ||||
|         <g id="Header" sketch:type="MSArtboardGroup" transform="translate(-327.000000, -12.000000)" fill="#FAFAFA"> | ||||
|             <g id="Search" sketch:type="MSLayerGroup" transform="translate(327.000000, 12.000000)"> | ||||
|                 <path d="M9.4369836,17.9549819 C13.9937934,17.9549819 17.6878176,14.2609577 17.6878176,9.70414786 C17.6878176,5.14733806 13.9937934,1.45331385 9.4369836,1.45331385 C4.8801738,1.45331385 1.18614959,5.14733806 1.18614959,9.70414786 C1.18614959,14.2609577 4.8801738,17.9549819 9.4369836,17.9549819 L9.4369836,17.9549819 L9.4369836,17.9549819 Z M9.4369836,14.9549819 C6.53702805,14.9549819 4.18614959,12.6041034 4.18614959,9.70414786 C4.18614959,6.80419231 6.53702805,4.45331385 9.4369836,4.45331385 C12.3369391,4.45331385 14.6878176,6.80419231 14.6878176,9.70414786 C14.6878176,12.6041034 12.3369391,14.9549819 9.4369836,14.9549819 Z M8.55898442,17.9088177 L6.03959979,24.4720391 C5.64616063,25.4969831 4.50444038,26.0120308 3.47355444,25.6163108 C2.44980588,25.2233305 1.93924273,24.0728244 2.33227833,23.0489317 L4.82760148,16.5483926 C5.91506785,17.2822068 7.18685775,17.7636774 8.55898442,17.9088177 L8.55898442,17.9088177 Z" id="Shape" sketch:type="MSShapeGroup"></path> | ||||
|             </g> | ||||
|         </g> | ||||
|     </g> | ||||
| </svg> | ||||
| After Width: | Height: | Size: 1.7 KiB | 
| @@ -3,7 +3,7 @@ | ||||
| <head> | ||||
|   <meta charset="UTF-8"> | ||||
|   <title>Hawk</title> | ||||
|   <meta name="viewport" content="width=device-width"> | ||||
|   <meta name="viewport" content="width=device-width, user-scalable=no, initial-scale=1"> | ||||
|  | ||||
|   <link rel='stylesheet' href='style.css' /> | ||||
| </head> | ||||
|   | ||||
							
								
								
									
										13678
									
								
								build/main.js
									
									
									
									
									
								
							
							
						
						
									
										13678
									
								
								build/main.js
									
									
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										171
									
								
								build/style.css
									
									
									
									
									
								
							
							
						
						
									
										171
									
								
								build/style.css
									
									
									
									
									
								
							| @@ -43,6 +43,22 @@ | ||||
|   width: 6px; | ||||
|   height: 24px; | ||||
| } | ||||
| .icon-search { | ||||
|   display: block; | ||||
|   background: url(/img/Search.svg) no-repeat; | ||||
|   width: 19px; | ||||
|   height: 27px; | ||||
| } | ||||
| .icon-cross { | ||||
|   display: block; | ||||
|   background: url(/img/Plus.svg) no-repeat; | ||||
|   width: 24px; | ||||
|   height: 24px; | ||||
|   transform: rotate(45deg); | ||||
| } | ||||
| .icon-cross svg * { | ||||
|   fill: white; | ||||
| } | ||||
| .regular-medium { | ||||
|   font-weight: normal; | ||||
| } | ||||
| @@ -104,10 +120,8 @@ input { | ||||
|   font-weight: 200; | ||||
|   font-size: 1.6rem; | ||||
| } | ||||
| label { | ||||
|   clear: left; | ||||
| } | ||||
| label::after { | ||||
| input[type='checkbox'] + label::after, | ||||
| input[type='radio'] + label::after { | ||||
|   content: ''; | ||||
|   display: block; | ||||
|   float: right; | ||||
| @@ -118,14 +132,36 @@ label::after { | ||||
|   background: transparent; | ||||
|   border: 1px solid #9b9b93; | ||||
| } | ||||
| input[type='checkbox'] { | ||||
|   clear: right; | ||||
|   float: right; | ||||
| input[type='checkbox'], | ||||
| input[type='radio'] { | ||||
|   display: none; | ||||
| } | ||||
| input:checked + label::after { | ||||
|   background: #63b0cd; | ||||
| } | ||||
| .coming-soon::after { | ||||
|   content: 'soon...'; | ||||
|   background: #f7c59f; | ||||
|   color: #39393a; | ||||
|   padding: 2px 8px; | ||||
|   border-radius: 12px; | ||||
|   font-size: 11px; | ||||
|   font-weight: normal; | ||||
| } | ||||
| li.coming-soon::after { | ||||
|   margin-right: 13px; | ||||
|   float: right; | ||||
| } | ||||
| button.coming-soon { | ||||
|   position: relative; | ||||
| } | ||||
| button.coming-soon::after { | ||||
|   position: absolute; | ||||
|   left: 50%; | ||||
|   top: 50%; | ||||
|   opacity: 0.8; | ||||
|   transform: translate(-50%, -50%) rotate(-45deg); | ||||
| } | ||||
| .file, | ||||
| .directory { | ||||
|   display: flex; | ||||
| @@ -141,11 +177,13 @@ input:checked + label::after { | ||||
| .file p, | ||||
| .directory p { | ||||
|   flex: 1 1; | ||||
|   text-overflow: ellipsis; | ||||
| } | ||||
| .file > span, | ||||
| .directory > span { | ||||
|   font-weight: 100; | ||||
|   font-size: 1.5rem; | ||||
|   margin-left: 1rem; | ||||
| } | ||||
| .file i, | ||||
| .directory i { | ||||
| @@ -176,6 +214,10 @@ header { | ||||
| } | ||||
| header h1 { | ||||
|   margin-left: -3rem; | ||||
|   flex: 1; | ||||
| } | ||||
| header i { | ||||
|   margin-right: 16px; | ||||
| } | ||||
| header button { | ||||
|   background: none; | ||||
| @@ -202,7 +244,7 @@ header button::before { | ||||
|   border-radius: 4px; | ||||
|   pointer-events: none; | ||||
|   opacity: 0; | ||||
|   transition: opacity 0.5s ease; | ||||
|   transition: opacity 0.3s ease; | ||||
|   box-shadow: 0 8px 16px 3px rgba(0, 0, 0, 0.2); | ||||
| } | ||||
| .menu.active { | ||||
| @@ -221,6 +263,10 @@ header button::before { | ||||
| .menu li:last-of-type { | ||||
|   border-bottom: none; | ||||
| } | ||||
| .menu li.disabled { | ||||
|   color: #9b9b93; | ||||
|   pointer-events: none; | ||||
| } | ||||
| nav { | ||||
|   display: flex; | ||||
|   flex-flow: column; | ||||
| @@ -229,18 +275,22 @@ nav { | ||||
|   top: 0; | ||||
|   width: 70vw; | ||||
|   height: 100vh; | ||||
|   overflow-y: auto; | ||||
|   background: #39393a; | ||||
|   color: white; | ||||
|   box-shadow: 3px 0 16px 5px rgba(0, 0, 0, 0.2); | ||||
|   box-shadow: 3px 0 16px 5px transparent, 0 0 0 1000px rgba(0, 0, 0, 0); | ||||
|   z-index: 6; | ||||
|   transition: left 0.5s ease; | ||||
|   transition: left 0.3s ease, box-shadow 0.3s ease; | ||||
| } | ||||
| nav ul:last-of-type li:last-of-type { | ||||
|   margin-bottom: 13px; | ||||
| } | ||||
| nav.active { | ||||
|   left: 0; | ||||
|   box-shadow: 3px 0 16px 5px rgba(0, 0, 0, 0.2), 0 0 0 1000px rgba(0, 0, 0, 0.55); | ||||
| } | ||||
| nav.active i { | ||||
|   pointer-events: all; | ||||
|   opacity: 0.99; | ||||
| } | ||||
| nav p { | ||||
|   margin-left: 1.6rem; | ||||
| @@ -252,6 +302,7 @@ nav ul { | ||||
|   padding-left: 0; | ||||
| } | ||||
| nav li { | ||||
|   display: flex; | ||||
|   font-weight: 200; | ||||
|   font-size: 1.6rem; | ||||
|   padding: 1rem 0 1rem 3rem; | ||||
| @@ -264,18 +315,21 @@ nav li:last-of-type { | ||||
|   padding-bottom: 0; | ||||
|   border-bottom: none; | ||||
| } | ||||
| nav li label { | ||||
|   flex: 1; | ||||
|   order: 0; | ||||
| } | ||||
| nav li input { | ||||
|   order: 1; | ||||
| } | ||||
| nav i { | ||||
|   display: block; | ||||
|   position: fixed; | ||||
|   left: 0; | ||||
|   left: 70vw; | ||||
|   top: 0; | ||||
|   pointer-events: none; | ||||
|   width: 100vw; | ||||
|   width: 30vw; | ||||
|   height: 100vh; | ||||
|   background: rgba(0, 0, 0, 0.55); | ||||
|   opacity: 0; | ||||
|   z-index: -1; | ||||
|   transition: opacity 0.5s ease; | ||||
|   pointer-events: none; | ||||
| } | ||||
| .toolbar { | ||||
|   display: flex; | ||||
| @@ -293,13 +347,20 @@ nav i { | ||||
|   flex: 1; | ||||
|   align-items: center; | ||||
|   width: 100vw; | ||||
|   height: 3.5rem; | ||||
|   height: 4.5rem; | ||||
|   overflow-x: auto; | ||||
|   padding: 8px; | ||||
|   box-sizing: border-box; | ||||
|   font-weight: 200; | ||||
|   font-size: 1.6rem; | ||||
|   background: #f8f8f8; | ||||
|   border-bottom: 1px solid rgba(0, 0, 0, 0.2); | ||||
|   overflow-x: scroll; | ||||
|   overflow-y: hidden; | ||||
|   white-space: nowrap; | ||||
| } | ||||
| .breadcrumb span { | ||||
|   white-space: nowrap; | ||||
| } | ||||
| .breadcrumb i { | ||||
|   margin: 0 2px; | ||||
| @@ -308,7 +369,7 @@ nav i { | ||||
|   color: #9b9b93; | ||||
| } | ||||
| .file-list { | ||||
|   height: calc(100vh - 13.5rem); | ||||
|   height: calc(100vh - 14.5rem); | ||||
|   overflow-x: hidden; | ||||
|   overflow-y: auto; | ||||
| } | ||||
| @@ -319,13 +380,13 @@ nav i { | ||||
|   top: 50%; | ||||
|   left: 50%; | ||||
|   transform: translate(-50%, -50%); | ||||
|   width: 335px; | ||||
|   width: 90vw; | ||||
|   height: auto; | ||||
|   padding: 1.5rem 1.7rem; | ||||
|   background: white; | ||||
|   box-shadow: 0 15px 24px 6px rgba(0, 0, 0, 0.2); | ||||
|   z-index: 3; | ||||
|   transition: opacity 0.5s ease; | ||||
|   z-index: 5; | ||||
|   transition: opacity 0.3s ease; | ||||
|   opacity: 0; | ||||
|   pointer-events: none; | ||||
| } | ||||
| @@ -356,6 +417,66 @@ nav i { | ||||
| .dialog .foot button:last-of-type { | ||||
|   margin-right: 0; | ||||
| } | ||||
| .sk-cube-grid { | ||||
|   opacity: 0; | ||||
|   pointer-events: none; | ||||
|   width: 4rem; | ||||
|   height: 4rem; | ||||
|   position: fixed; | ||||
|   left: 50%; | ||||
|   top: 50%; | ||||
|   margin-top: -2rem; | ||||
|   margin-left: -2rem; | ||||
|   z-index: 5; | ||||
|   transition: opacity 0.3s ease; | ||||
| } | ||||
| .sk-cube-grid.show { | ||||
|   opacity: 1; | ||||
| } | ||||
| .sk-cube-grid .sk-cube { | ||||
|   width: 33.33%; | ||||
|   height: 33.33%; | ||||
|   background-color: #333; | ||||
|   float: left; | ||||
|   animation: sk-cubeGridScaleDelay 1.3s infinite ease-in-out; | ||||
| } | ||||
| .sk-cube-grid .sk-cube1 { | ||||
|   animation-delay: 0.2s; | ||||
| } | ||||
| .sk-cube-grid .sk-cube2 { | ||||
|   animation-delay: 0.3s; | ||||
| } | ||||
| .sk-cube-grid .sk-cube3 { | ||||
|   animation-delay: 0.4s; | ||||
| } | ||||
| .sk-cube-grid .sk-cube4 { | ||||
|   animation-delay: 0.1s; | ||||
| } | ||||
| .sk-cube-grid .sk-cube5 { | ||||
|   animation-delay: 0.2s; | ||||
| } | ||||
| .sk-cube-grid .sk-cube6 { | ||||
|   animation-delay: 0.3s; | ||||
| } | ||||
| .sk-cube-grid .sk-cube7 { | ||||
|   animation-delay: 0s; | ||||
| } | ||||
| .sk-cube-grid .sk-cube8 { | ||||
|   animation-delay: 0.1s; | ||||
| } | ||||
| .sk-cube-grid .sk-cube9 { | ||||
|   animation-delay: 0.2s; | ||||
| } | ||||
| @keyframes sk-cubeGridScaleDelay { | ||||
|   0%, | ||||
|   70%, | ||||
|   100% { | ||||
|     transform: scale3D(1, 1, 1); | ||||
|   } | ||||
|   35% { | ||||
|     transform: scale3D(0, 0, 1); | ||||
|   } | ||||
| } | ||||
| html, | ||||
| body { | ||||
|   margin: 0; | ||||
| @@ -372,3 +493,7 @@ body { | ||||
|   display: flex; | ||||
|   flex-flow: column; | ||||
| } | ||||
| a { | ||||
|   color: currentColor; | ||||
|   text-decoration: none; | ||||
| } | ||||
|   | ||||
										
											Binary file not shown.
										
									
								
							| @@ -43,6 +43,8 @@ | ||||
|     "grunt-contrib-watch": "^0.6.1", | ||||
|     "grunt-fxos": "^0.1.2", | ||||
|     "grunt-task-loader": "^0.6.0", | ||||
|     "hammerjs": "^2.0.4", | ||||
|     "immutable": "^3.7.5", | ||||
|     "less-plugin-clean-css": "^1.5.1", | ||||
|     "lodash": "^3.10.1", | ||||
|     "react": "^0.13.3", | ||||
|   | ||||
							
								
								
									
										
											BIN
										
									
								
								src/img/Search.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								src/img/Search.png
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 690 B | 
							
								
								
									
										14
									
								
								src/img/Search.svg
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										14
									
								
								src/img/Search.svg
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,14 @@ | ||||
| <?xml version="1.0" encoding="UTF-8" standalone="no"?> | ||||
| <svg width="19px" height="27px" viewBox="0 0 19 27" version="1.1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" xmlns:sketch="http://www.bohemiancoding.com/sketch/ns"> | ||||
|     <!-- Generator: Sketch 3.3.3 (12072) - http://www.bohemiancoding.com/sketch --> | ||||
|     <title>Search</title> | ||||
|     <desc>Created with Sketch.</desc> | ||||
|     <defs></defs> | ||||
|     <g id="Components" stroke="none" stroke-width="1" fill="none" fill-rule="evenodd" sketch:type="MSPage"> | ||||
|         <g id="Header" sketch:type="MSArtboardGroup" transform="translate(-327.000000, -12.000000)" fill="#FAFAFA"> | ||||
|             <g id="Search" sketch:type="MSLayerGroup" transform="translate(327.000000, 12.000000)"> | ||||
|                 <path d="M9.4369836,17.9549819 C13.9937934,17.9549819 17.6878176,14.2609577 17.6878176,9.70414786 C17.6878176,5.14733806 13.9937934,1.45331385 9.4369836,1.45331385 C4.8801738,1.45331385 1.18614959,5.14733806 1.18614959,9.70414786 C1.18614959,14.2609577 4.8801738,17.9549819 9.4369836,17.9549819 L9.4369836,17.9549819 L9.4369836,17.9549819 Z M9.4369836,14.9549819 C6.53702805,14.9549819 4.18614959,12.6041034 4.18614959,9.70414786 C4.18614959,6.80419231 6.53702805,4.45331385 9.4369836,4.45331385 C12.3369391,4.45331385 14.6878176,6.80419231 14.6878176,9.70414786 C14.6878176,12.6041034 12.3369391,14.9549819 9.4369836,14.9549819 Z M8.55898442,17.9088177 L6.03959979,24.4720391 C5.64616063,25.4969831 4.50444038,26.0120308 3.47355444,25.6163108 C2.44980588,25.2233305 1.93924273,24.0728244 2.33227833,23.0489317 L4.82760148,16.5483926 C5.91506785,17.2822068 7.18685775,17.7636774 8.55898442,17.9088177 L8.55898442,17.9088177 Z" id="Shape" sketch:type="MSShapeGroup"></path> | ||||
|             </g> | ||||
|         </g> | ||||
|     </g> | ||||
| </svg> | ||||
| After Width: | Height: | Size: 1.7 KiB | 
| @@ -3,7 +3,7 @@ | ||||
| <head> | ||||
|   <meta charset="UTF-8"> | ||||
|   <title>Hawk</title> | ||||
|   <meta name="viewport" content="width=device-width"> | ||||
|   <meta name="viewport" content="width=device-width, user-scalable=no, initial-scale=1"> | ||||
|  | ||||
|   <link rel='stylesheet' href='style.css' /> | ||||
| </head> | ||||
|   | ||||
| @@ -1,4 +1,4 @@ | ||||
| import { CREATE_FILE, SHARE_FILE, RENAME_FILE, ACTIVE_FILE, DELETE_FILE } from 'actions/types'; | ||||
| import { CREATE_FILE, SHARE_FILE, RENAME_FILE, ACTIVE_FILE, DELETE_FILE, MOVE_FILE, COPY_FILE } from 'actions/types'; | ||||
|  | ||||
| export function create(path, directory = false) { | ||||
|   return { | ||||
| @@ -21,6 +21,20 @@ export function rename(file, name) { | ||||
|   } | ||||
| } | ||||
|  | ||||
| export function move(file, newPath) { | ||||
|   return { | ||||
|     type: MOVE_FILE, | ||||
|     file, newPath | ||||
|   } | ||||
| } | ||||
|  | ||||
| export function copy(file, newPath) { | ||||
|   return { | ||||
|     type: COPY_FILE, | ||||
|     file, newPath | ||||
|   } | ||||
| } | ||||
|  | ||||
| export function active(file = null) { | ||||
|   return { | ||||
|     type: ACTIVE_FILE, | ||||
| @@ -28,8 +42,7 @@ export function active(file = null) { | ||||
|   } | ||||
| } | ||||
|  | ||||
| export function deleteFile(file) { | ||||
|   console.log('constructing deleteFile action', file); | ||||
| export function remove(file) { | ||||
|   return { | ||||
|     type: DELETE_FILE, | ||||
|     file | ||||
|   | ||||
| @@ -1,6 +1,13 @@ | ||||
| import { LIST_FILES, FILES_VIEW, SELECT_VIEW, REFRESH } from 'actions/types'; | ||||
| import { LIST_FILES, FILES_VIEW, SELECT_VIEW, REFRESH, SEARCH } from 'actions/types'; | ||||
| import store from 'store'; | ||||
|  | ||||
| export function listFiles(files) { | ||||
|   return { | ||||
|     type: LIST_FILES, | ||||
|     files | ||||
|   }; | ||||
| } | ||||
|  | ||||
| export function refresh() { | ||||
|   return { | ||||
|     type: REFRESH | ||||
| @@ -34,3 +41,10 @@ export function selectView(active = true) { | ||||
|     active | ||||
|   } | ||||
| } | ||||
|  | ||||
| export function search(keywords) { | ||||
|   return { | ||||
|     type: SEARCH, | ||||
|     keywords | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -1,8 +0,0 @@ | ||||
| import { LIST_FILES } from 'actions/types'; | ||||
|  | ||||
| export default function listFiles(files) { | ||||
|   return { | ||||
|     type: LIST_FILES, | ||||
|     files | ||||
|   }; | ||||
| } | ||||
							
								
								
									
										22
									
								
								src/js/actions/spinner.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										22
									
								
								src/js/actions/spinner.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,22 @@ | ||||
| import { SPINNER } from 'actions/types'; | ||||
|  | ||||
| export function show() { | ||||
|   return { | ||||
|     type: SPINNER, | ||||
|     active: true | ||||
|   } | ||||
| } | ||||
|  | ||||
| export function hide() { | ||||
|   return { | ||||
|     type: SPINNER, | ||||
|     active: false | ||||
|   } | ||||
| } | ||||
|  | ||||
| export function toggle() { | ||||
|   return { | ||||
|     type: SPINNER, | ||||
|     active: 'toggle' | ||||
|   } | ||||
| } | ||||
| @@ -16,11 +16,15 @@ const TYPES = { | ||||
|   RENAME_FILE: Symbol('RENAME_FILE'), | ||||
|   ACTIVE_FILE: Symbol('ACTIVE_FILE'), | ||||
|   DELETE_FILE: Symbol('DELETE_FILE'), | ||||
|   COPY_FILE: Symbol('COPY_FILE'), | ||||
|   MOVE_FILE: Symbol('MOVE_FILE'), | ||||
|  | ||||
|   MENU: Symbol('MENU'), | ||||
|  | ||||
|   DIALOG: Symbol('DIALOG'), | ||||
|  | ||||
|   SPINNER: Symbol('SPINNER'), | ||||
|  | ||||
|   SETTINGS: Symbol('SETTINGS'), | ||||
|  | ||||
|   SEARCH: Symbol('SEARCH') | ||||
|   | ||||
| @@ -75,16 +75,25 @@ export async function createDirectory(...args) { | ||||
|   return parent.createDirectory(...args); | ||||
| } | ||||
|  | ||||
| export async function remove(file) { | ||||
| export async function remove(file, deep) { | ||||
|   let parent = await root(); | ||||
|  | ||||
|   return parent.remove(file); | ||||
|   console.log(deep); | ||||
|   return parent[deep ? 'removeDeep' : 'remove'](file); | ||||
| } | ||||
|  | ||||
| export async function move(file, newPath) { | ||||
|   let path = (file.path || '').replace(/^\//, ''); // remove starting slash | ||||
|   let oldPath = path + file.name; | ||||
|  | ||||
|   let process = await copy(file, newPath); | ||||
|   return remove(oldPath, true); | ||||
| } | ||||
|  | ||||
| export async function copy(file, newPath) { | ||||
|   let path = (file.path || '').replace(/^\//, ''); // remove starting slash | ||||
|   let oldPath = path + file.name; | ||||
|  | ||||
|   newPath = newPath.replace(/^\//, ''); | ||||
|  | ||||
|   let target = await getFile(oldPath); | ||||
| @@ -102,7 +111,6 @@ export async function move(file, newPath) { | ||||
|       await move(child, newPath + '/' + child.name); | ||||
|     } | ||||
|  | ||||
|     await parent.remove(oldPath); | ||||
|     return; | ||||
|   } else { | ||||
|     let content = await readFile(oldPath); | ||||
| @@ -114,6 +122,6 @@ export async function move(file, newPath) { | ||||
|       request.onsuccess = resolve; | ||||
|       request.onerror = reject; | ||||
|       request.onabort = reject; | ||||
|     }).then(() => sdcard().delete(oldPath)); | ||||
|     }); | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -40,7 +40,9 @@ export default class Breadcrumb extends Component { | ||||
|  | ||||
|     return ( | ||||
|       <div className='breadcrumb'> | ||||
|         {els} | ||||
|         <div> | ||||
|           {els} | ||||
|         </div> | ||||
|       </div> | ||||
|     ); | ||||
|   } | ||||
|   | ||||
| @@ -4,10 +4,16 @@ import { show } from 'actions/menu'; | ||||
| import { active } from 'actions/file'; | ||||
| import { MENU_WIDTH } from './menu'; | ||||
| import store from 'store'; | ||||
| import entry from './mixins/entry'; | ||||
|  | ||||
| const MENU_TOP_SPACE = 20; | ||||
|  | ||||
| export default class Directory extends Component { | ||||
|   constructor() { | ||||
|     super(); | ||||
|     Object.assign(this, entry); | ||||
|   } | ||||
|  | ||||
|   render() { | ||||
|     let checkId = `file-${this.props.index}`; | ||||
|  | ||||
| @@ -40,28 +46,4 @@ export default class Directory extends Component { | ||||
|  | ||||
|     store.dispatch(changedir(file.path.slice(1) + file.name)); | ||||
|   } | ||||
|  | ||||
|   contextMenu(e) { | ||||
|     e.preventDefault(); | ||||
|  | ||||
|     let rect = React.findDOMNode(this.refs.container).getBoundingClientRect(); | ||||
|     let {x, y, width, height} = rect; | ||||
|  | ||||
|     let left = x + width / 2 - MENU_WIDTH / 2, | ||||
|         top  = y + height / 2 + MENU_TOP_SPACE; | ||||
|     store.dispatch(show('directoryMenu', {style: {left, top}})); | ||||
|     store.dispatch(active(this.props.index)); | ||||
|   } | ||||
|  | ||||
|   select() { | ||||
|     let current = (store.getState().get('activeFile') || []).slice(0); | ||||
|     let index = this.props.index; | ||||
|  | ||||
|     if (current.indexOf(index) > -1) { | ||||
|       current.splice(current.indexOf(index), 1); | ||||
|     } else { | ||||
|       current.push(index) | ||||
|     } | ||||
|     store.dispatch(active(current)); | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -4,6 +4,8 @@ import File from './file'; | ||||
| import Directory from './directory'; | ||||
| import store from 'store'; | ||||
| import { type } from 'utils'; | ||||
| import Hammer from 'hammerjs'; | ||||
| import changedir from 'actions/changedir'; | ||||
|  | ||||
| @connect(props) | ||||
| export default class FileList extends Component { | ||||
| @@ -17,7 +19,7 @@ export default class FileList extends Component { | ||||
|     let settings = store.getState().get('settings'); | ||||
|  | ||||
|     let els = files.map((file, index) => { | ||||
|       let selected = activeFile.indexOf(index) > -1; | ||||
|       let selected = activeFile.length && activeFile.indexOf(file) > -1; | ||||
|       if (type(file) === 'File') { | ||||
|         return <File selectView={selectView} selected={selected} key={index} index={index} name={file.name} size={file.size} />; | ||||
|       } else { | ||||
| @@ -26,11 +28,25 @@ export default class FileList extends Component { | ||||
|     }); | ||||
|  | ||||
|     return ( | ||||
|       <div className='file-list'> | ||||
|       <div className='file-list' ref='container'> | ||||
|         {els} | ||||
|       </div> | ||||
|     ); | ||||
|   } | ||||
|  | ||||
|   componentDidMount() { | ||||
|     let container = React.findDOMNode(this.refs.container); | ||||
|     let touch = Hammer(container); | ||||
|  | ||||
|     touch.on('swipe', e => { | ||||
|       let current = store.getState().get('cwd'); | ||||
|       let up = current.split('/').slice(0, -1).join('/'); | ||||
|  | ||||
|       if (up === current) return; | ||||
|  | ||||
|       store.dispatch(changedir(up)); | ||||
|     }).set({direction: Hammer.DIRECTION_RIGHT}); | ||||
|   } | ||||
| } | ||||
|  | ||||
| function props(state) { | ||||
|   | ||||
| @@ -4,12 +4,14 @@ import { active } from 'actions/file'; | ||||
| import { MENU_WIDTH } from './menu'; | ||||
| import store from 'store'; | ||||
| import { humanSize } from 'utils'; | ||||
| import entry from './mixins/entry'; | ||||
|  | ||||
| const MENU_TOP_SPACE = 20; | ||||
|  | ||||
| export default class File extends Component { | ||||
|   constructor() { | ||||
|     super(); | ||||
|     Object.assign(this, entry); | ||||
|   } | ||||
|  | ||||
|   render() { | ||||
| @@ -22,7 +24,7 @@ export default class File extends Component { | ||||
|     } | ||||
|  | ||||
|     let clickHandler = this.props.selectView ? this.select.bind(this) | ||||
|                                              : null; | ||||
|                                              : this.open.bind(this); | ||||
|  | ||||
|     return ( | ||||
|       <div className='file' ref='container' | ||||
| @@ -39,27 +41,16 @@ export default class File extends Component { | ||||
|     ); | ||||
|   } | ||||
|  | ||||
|   contextMenu(e) { | ||||
|     e.preventDefault(); | ||||
|   open(e) { | ||||
|     let file = store.getState().get('files')[this.props.index]; | ||||
|  | ||||
|     let rect = React.findDOMNode(this.refs.container).getBoundingClientRect(); | ||||
|     let {x, y, width, height} = rect; | ||||
|  | ||||
|     let left = x + width / 2 - MENU_WIDTH / 2, | ||||
|         top  = y + height / 2 + MENU_TOP_SPACE; | ||||
|     store.dispatch(show('fileMenu', {style: {left, top}})); | ||||
|     store.dispatch(active(this.props.index)); | ||||
|   } | ||||
|  | ||||
|   select() { | ||||
|     let current = (store.getState().get('activeFile') || []).slice(0); | ||||
|     let index = this.props.index; | ||||
|  | ||||
|     if (current.indexOf(index) > -1) { | ||||
|       current.splice(current.indexOf(index), 1); | ||||
|     } else { | ||||
|       current.push(index) | ||||
|     } | ||||
|     store.dispatch(active(current)); | ||||
|     let name = file.type === 'application/pdf' ? 'view' : 'open'; | ||||
|     new MozActivity({ | ||||
|       name, | ||||
|       data: { | ||||
|         type: file.type, | ||||
|         blob: file | ||||
|       } | ||||
|     }) | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -1,18 +1,34 @@ | ||||
| import React, { Component } from 'react'; | ||||
| import { toggle } from 'actions/navigation'; | ||||
| import store from 'store'; | ||||
| import { show } from 'actions/dialog'; | ||||
| import { search } from 'actions/files-view'; | ||||
| import { bind } from 'store'; | ||||
| import { connect } from 'react-redux'; | ||||
|  | ||||
| @connect(props) | ||||
| export default class Header extends Component { | ||||
|   render() { | ||||
|     let i; | ||||
|  | ||||
|     if (this.props.search) { | ||||
|       i = <i className='icon-cross' onClick={bind(search())} /> | ||||
|     } else { | ||||
|       i = <i className='icon-search' onClick={bind(show('searchDialog'))} /> | ||||
|     } | ||||
|  | ||||
|     return ( | ||||
|       <header> | ||||
|         <button className='drawer' onClick={this.toggleNavigation.bind(this)}></button> | ||||
|         <button className='drawer' onTouchStart={bind(toggle())} /> | ||||
|         <h1 className='regular-medium'>Hawk</h1> | ||||
|  | ||||
|         {i} | ||||
|       </header> | ||||
|     ); | ||||
|   } | ||||
| } | ||||
|  | ||||
|   toggleNavigation() { | ||||
|     store.dispatch(toggle()); | ||||
| function props(state) { | ||||
|   return { | ||||
|     search: state.get('search') | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -8,7 +8,10 @@ export default class Menu extends Component { | ||||
|     items = items || []; | ||||
|  | ||||
|     let els = items.map((item, index) => { | ||||
|       return <li key={index} onClick={item.action.bind(this)}>{item.name}</li> | ||||
|       let disabled = !(typeof item.enabled === 'function' ? item.enabled() : true) | ||||
|       let className = disabled ? 'disabled' : ''; | ||||
|  | ||||
|       return <li key={index} className={className} onClick={item.action.bind(this)}>{item.name}</li> | ||||
|     }); | ||||
|     let className = 'menu ' + (active ? 'active' : ''); | ||||
|  | ||||
|   | ||||
							
								
								
									
										26
									
								
								src/js/components/mixins/entry.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										26
									
								
								src/js/components/mixins/entry.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,26 @@ | ||||
| export default { | ||||
|   contextMenu(e) { | ||||
|     e.preventDefault(); | ||||
|  | ||||
|     let file = store.getState().get('files')[this.props.index]; | ||||
|     let rect = React.findDOMNode(this.refs.container).getBoundingClientRect(); | ||||
|     let {x, y, width, height} = rect; | ||||
|  | ||||
|     let left = x + width / 2 - MENU_WIDTH / 2, | ||||
|         top  = y + height / 2 + MENU_TOP_SPACE; | ||||
|     store.dispatch(show('fileMenu', {style: {left, top}})); | ||||
|     store.dispatch(active([file])); | ||||
|   }, | ||||
|  | ||||
|   select() { | ||||
|     let current = store.getState().get('activeFile').slice(0); | ||||
|     let file = store.getState().get('files')[this.props.index]; | ||||
|  | ||||
|     if (current.indexOf(file) > -1) { | ||||
|       current.splice(current.indexOf(file), 1); | ||||
|     } else { | ||||
|       current.push(file) | ||||
|     } | ||||
|     store.dispatch(active(current)); | ||||
|   } | ||||
| } | ||||
| @@ -1,9 +1,9 @@ | ||||
| import React, { Component } from 'react'; | ||||
| import { connect } from 'react-redux'; | ||||
| import { hide } from 'actions/navigation'; | ||||
| import { hide as hideNavigation } from 'actions/navigation'; | ||||
| import camelCase from 'lodash/string/camelCase'; | ||||
| import updateSettings from 'actions/settings'; | ||||
| import store from 'store'; | ||||
| import store, { bind } from 'store'; | ||||
|  | ||||
| @connect(props) | ||||
| export default class Navigation extends Component { | ||||
| @@ -11,23 +11,38 @@ export default class Navigation extends Component { | ||||
|     let { settings } = this.props; | ||||
|  | ||||
|     return ( | ||||
|       <nav className={this.props.active ? 'active' : ''}> | ||||
|         <i onClick={this.hide.bind(this)} /> | ||||
|       <nav className={this.props.active ? 'active' : ''} onChange={this.onChange.bind(this)}> | ||||
|         <i onTouchStart={this.hide} /> | ||||
|  | ||||
|         <p>Filter</p> | ||||
|         <ul> | ||||
|           <li>Picture</li> | ||||
|           <li>Video</li> | ||||
|           <li>Audio</li> | ||||
|           <li> | ||||
|             <input id='filter-all' name='filter' value='' type='radio' defaultChecked={!settings.filter} /> | ||||
|             <label htmlFor='filter-all'>All</label> | ||||
|           </li> | ||||
|           <li> | ||||
|             <input id='filter-image' name='filter' value='image' type='radio' defaultChecked={settings.filter === 'image'} /> | ||||
|             <label htmlFor='filter-image'>Image</label> | ||||
|           </li> | ||||
|           <li> | ||||
|             <input id='filter-video' name='filter' value='video' type='radio' defaultChecked={settings.filter === 'video'} /> | ||||
|             <label htmlFor='filter-video'>Video</label> | ||||
|           </li> | ||||
|           <li> | ||||
|             <input id='filter-audio' name='filter' value='audio' type='radio' defaultChecked={settings.filter === 'audio'} /> | ||||
|             <label htmlFor='filter-audio'>Audio</label> | ||||
|           </li> | ||||
|         </ul> | ||||
|  | ||||
|         <p>Tools</p> | ||||
|         <ul> | ||||
|           <li>FTP Browser</li> | ||||
|           <li className='coming-soon'> | ||||
|             <label>FTP Browser</label> | ||||
|           </li> | ||||
|         </ul> | ||||
|  | ||||
|         <p>Preferences</p> | ||||
|         <ul onChange={this.onChange.bind(this)}> | ||||
|         <ul> | ||||
|           <li> | ||||
|             <input type='checkbox' id='showHiddenFiles' defaultChecked={settings.showHiddenFiles} /> | ||||
|             <label htmlFor='showHiddenFiles'>Show Hidden Files</label> | ||||
| @@ -36,28 +51,47 @@ export default class Navigation extends Component { | ||||
|             <input id='showDirectoriesFirst' type='checkbox' defaultChecked={settings.showDirectoriesFirst} /> | ||||
|             <label htmlFor='showDirectoriesFirst'>Show Directories First</label> | ||||
|           </li> | ||||
|           <li>Advanced Preferences</li> | ||||
|           <li className='coming-soon'> | ||||
|             <label>Advanced Preferences</label> | ||||
|           </li> | ||||
|         </ul> | ||||
|  | ||||
|         <p>External</p> | ||||
|         <ul> | ||||
|           <li> | ||||
|             <label><a href='https://github.com/mdibaiee/Hawk'>GitHub</a></label> | ||||
|           </li> | ||||
|           <li> | ||||
|             <label><a href='https://github.com/mdibaiee/Hawk/issues'>Report Bugs</a></label> | ||||
|           </li> | ||||
|           <li> | ||||
|             <label><a href='http://dibaiee.ir/Hawk'>Website</a></label> | ||||
|           </li> | ||||
|           <li> | ||||
|             <label><a href='http://dibaiee.ir'>Mahdi Dibaiee</a></label> | ||||
|           </li> | ||||
|         </ul> | ||||
|       </nav> | ||||
|     ); | ||||
|   } | ||||
|  | ||||
|   hide() { | ||||
|     this.props.dispatch(hide()); | ||||
|   } | ||||
|  | ||||
|   onChange(e) { | ||||
|     if (e.target.nodeName.toLowerCase() !== 'input') return; | ||||
|  | ||||
|     let key = e.target.id; | ||||
|     let value = this.props.settings[key]; | ||||
|     let key = e.target.name || e.target.id; | ||||
|     let value = e.target.value === undefined ? e.target.checked : e.target.value; | ||||
|  | ||||
|     let action = updateSettings({ | ||||
|       [key]: e.target.checked | ||||
|       [key]: value | ||||
|     }); | ||||
|  | ||||
|     store.dispatch(action); | ||||
|   } | ||||
|  | ||||
|   hide(e) { | ||||
|     e.preventDefault(); | ||||
|     e.stopPropagation(); | ||||
|  | ||||
|     store.dispatch(hideNavigation()); | ||||
|   } | ||||
| } | ||||
|  | ||||
| function props(store) { | ||||
|   | ||||
| @@ -6,6 +6,7 @@ import Breadcrumb from 'components/breadcrumb'; | ||||
| import Toolbar from 'components/toolbar'; | ||||
| import Menu from 'components/menu'; | ||||
| import Dialog from 'components/dialog'; | ||||
| import Spinner from 'components/spinner'; | ||||
| import { connect } from 'react-redux'; | ||||
| import { hideAll as hideAllMenus } from 'actions/menu'; | ||||
| import { hideAll as hideAllDialogs} from 'actions/dialog'; | ||||
| @@ -24,11 +25,12 @@ let RenameDialog = connect(state => state.get('renameDialog'))(Dialog); | ||||
| let DeleteDialog = connect(state => state.get('deleteDialog'))(Dialog); | ||||
| let ErrorDialog = connect(state => state.get('errorDialog'))(Dialog); | ||||
| let CreateDialog = connect(state => state.get('createDialog'))(Dialog); | ||||
| let SearchDialog = connect(state => state.get('searchDialog'))(Dialog); | ||||
|  | ||||
| export default class Root extends Component { | ||||
|   render() { | ||||
|     return ( | ||||
|       <div onTouchStart={this.touchStart.bind(this)}> | ||||
|       <div onTouchStart={this.touchStart.bind(this)} onClick={this.onClick.bind(this)}> | ||||
|         <Header /> | ||||
|         <Breadcrumb /> | ||||
|         <Navigation /> | ||||
| @@ -43,19 +45,46 @@ export default class Root extends Component { | ||||
|         <DeleteDialog /> | ||||
|         <ErrorDialog /> | ||||
|         <CreateDialog /> | ||||
|         <SearchDialog /> | ||||
|  | ||||
|         <Spinner /> | ||||
|       </div> | ||||
|     ); | ||||
|   } | ||||
|  | ||||
|   touchStart(e) { | ||||
|     let active = document.querySelector('.active'); | ||||
|     let inside = e.target.closest('.menu') || e.target.closest('.dialog'); | ||||
|     if (!inside && active) { | ||||
|     let inside = e.target.closest('.active'); | ||||
|     if (active && !inside) { | ||||
|       e.preventDefault(); | ||||
|       e.stopPropagation(); | ||||
|  | ||||
|       store.dispatch(hideAllMenus()); | ||||
|       store.dispatch(hideAllDialogs()); | ||||
|     } | ||||
|  | ||||
|     if (document.querySelector('.sk-cube-grid.show')) { | ||||
|       e.preventDefault(); | ||||
|       e.stopPropagation(); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   onClick(e) { | ||||
|     let tag = e.target.nodeName.toLowerCase(); | ||||
|     if (tag === 'a') { | ||||
|       let url = new URL(e.target.href); | ||||
|  | ||||
|       if (url.origin !== location.origin) { | ||||
|         e.preventDefault(); | ||||
|         new MozActivity({ | ||||
|           name: 'view', | ||||
|  | ||||
|           data: { | ||||
|             type: 'url', | ||||
|             url: e.target.href | ||||
|           } | ||||
|         }) | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| } | ||||
|   | ||||
							
								
								
									
										28
									
								
								src/js/components/spinner.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										28
									
								
								src/js/components/spinner.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,28 @@ | ||||
| import React, { Component } from 'react'; | ||||
| import { connect } from 'react-redux'; | ||||
|  | ||||
| @connect(props) | ||||
| export default class Spinner extends Component { | ||||
|   render() { | ||||
|     let className = 'sk-cube-grid ' + (this.props.active ? ' show' : ''); | ||||
|     return ( | ||||
|       <div className={className}> | ||||
|         <div className="sk-cube sk-cube1"></div> | ||||
|         <div className="sk-cube sk-cube2"></div> | ||||
|         <div className="sk-cube sk-cube3"></div> | ||||
|         <div className="sk-cube sk-cube4"></div> | ||||
|         <div className="sk-cube sk-cube5"></div> | ||||
|         <div className="sk-cube sk-cube6"></div> | ||||
|         <div className="sk-cube sk-cube7"></div> | ||||
|         <div className="sk-cube sk-cube8"></div> | ||||
|         <div className="sk-cube sk-cube9"></div> | ||||
|       </div> | ||||
|     ); | ||||
|   } | ||||
| } | ||||
|  | ||||
| function props(state) { | ||||
|   return { | ||||
|     active: state.get('spinner') | ||||
|   } | ||||
| } | ||||
| @@ -10,7 +10,7 @@ export default class Toolbar extends Component { | ||||
|     return ( | ||||
|       <div className='toolbar'> | ||||
|         <button className='icon-plus' onClick={this.newFile} /> | ||||
|         <button className='icon-view' onClick={bind(toggleView())} /> | ||||
|         <button className='icon-view coming-soon' onClick={bind(toggleView())} /> | ||||
|         <button className='icon-refresh' onClick={bind(refresh())} /> | ||||
|         <button className='icon-select' onClick={bind(selectView('toggle'))} /> | ||||
|         <button className='icon-more' onClick={this.showMore.bind(this)} ref='more' /> | ||||
|   | ||||
| @@ -1,6 +1,7 @@ | ||||
| import React from 'react'; | ||||
| import { hide, hideAll } from 'actions/dialog'; | ||||
| import { rename, deleteFile, create, active } from 'actions/file'; | ||||
| import { rename, remove, create, active } from 'actions/file'; | ||||
| import { search } from 'actions/files-view'; | ||||
| import store, { bind } from 'store'; | ||||
|  | ||||
| export default { | ||||
| @@ -70,7 +71,7 @@ export default { | ||||
|         text: 'Yes', | ||||
|         action() { | ||||
|           let activeFile = store.getState().get('activeFile'); | ||||
|           this.props.dispatch(deleteFile(activeFile)); | ||||
|           this.props.dispatch(remove(activeFile)); | ||||
|           this.props.dispatch(hideAll()); | ||||
|           this.props.dispatch(active()); | ||||
|         }, | ||||
| @@ -84,5 +85,26 @@ export default { | ||||
|       text: 'Continue', | ||||
|       action: bind(hideAll()) | ||||
|     }] | ||||
|   }, | ||||
|   searchDialog: { | ||||
|     title: 'Search', | ||||
|     description: 'Enter keywords to search for', | ||||
|     input: true, | ||||
|     buttons: [ | ||||
|       { | ||||
|         text: 'Cancel', | ||||
|         action: bind(hideAll()) | ||||
|       }, | ||||
|       { | ||||
|         text: 'Search', | ||||
|         action() { | ||||
|           let input = React.findDOMNode(this.refs.input); | ||||
|  | ||||
|           let action = search(input.value); | ||||
|           this.props.dispatch(action); | ||||
|           this.props.dispatch(hideAll()); | ||||
|         } | ||||
|       } | ||||
|     ] | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -1,5 +1,7 @@ | ||||
| import { hideAll } from 'actions/menu'; | ||||
| import { show } from 'actions/dialog'; | ||||
| import { selectView } from 'actions/files-view'; | ||||
| import { copy, move } from 'actions/file'; | ||||
| import store from 'store'; | ||||
|  | ||||
| const entryMenu = { | ||||
| @@ -9,8 +11,7 @@ const entryMenu = { | ||||
|       action() { | ||||
|         let files = store.getState().get('files'); | ||||
|         let active = store.getState().get('activeFile'); | ||||
|         const name = files[active].name; | ||||
|         const description = `Are you sure you want to remove ${name}?`; | ||||
|         const description = `Enter the new name for ${active[0].name}?`; | ||||
|  | ||||
|         store.dispatch(hideAll()); | ||||
|         store.dispatch(show('renameDialog', {description})); | ||||
| @@ -21,11 +22,17 @@ const entryMenu = { | ||||
|       action() { | ||||
|         let files = store.getState().get('files'); | ||||
|         let active = store.getState().get('activeFile'); | ||||
|         const name = files[active].name; | ||||
|         const description = `Are you sure you want to remove ${name}?`; | ||||
|         const description = `Are you sure you want to remove ${active[0].name}?`; | ||||
|         store.dispatch(hideAll()); | ||||
|         store.dispatch(show('deleteDialog', {description})); | ||||
|       } | ||||
|     }, | ||||
|     { | ||||
|       name: 'Copy', | ||||
|       action() { | ||||
|         store.dispatch(selectView(false)); | ||||
|         store.dispatch(hideAll()); | ||||
|       } | ||||
|     } | ||||
|   ] | ||||
| }; | ||||
| @@ -39,16 +46,55 @@ const moreMenu = { | ||||
|         let active = store.getState().get('activeFile'); | ||||
|  | ||||
|         let description; | ||||
|         if (active.length) { | ||||
|         if (active.length > 1) { | ||||
|           const count = active.length; | ||||
|           description = `Are you sure you want to remove ${count} files?`; | ||||
|         } else { | ||||
|           const name = files[active].name; | ||||
|           const name = active[0].name; | ||||
|           description = `Are you sure you want to remove ${name}?`; | ||||
|         } | ||||
|  | ||||
|         store.dispatch(hideAll()); | ||||
|         store.dispatch(show('deleteDialog', {description})); | ||||
|       }, | ||||
|       enabled() { | ||||
|         return store.getState().get('activeFile'); | ||||
|       } | ||||
|     }, | ||||
|     { | ||||
|       name: 'Copy', | ||||
|       action() { | ||||
|         store.dispatch(selectView(false)); | ||||
|         store.dispatch(hideAll()); | ||||
|       }, | ||||
|       enabled() { | ||||
|         return store.getState().get('activeFile'); | ||||
|       } | ||||
|     }, | ||||
|     { | ||||
|       name: 'Paste', | ||||
|       enabled() { | ||||
|         return store.getState().get('activeFile'); | ||||
|       }, | ||||
|       action() { | ||||
|         let active = store.getState().get('activeFile'); | ||||
|         let cwd = store.getState().get('cwd'); | ||||
|  | ||||
|         store.dispatch(copy(active, cwd)); | ||||
|         store.dispatch(hideAll()); | ||||
|       } | ||||
|     }, | ||||
|     { | ||||
|       name: 'Move', | ||||
|       enabled() { | ||||
|         return store.getState().get('activeFile'); | ||||
|       }, | ||||
|       action() { | ||||
|         let active = store.getState().get('activeFile'); | ||||
|         let cwd = store.getState().get('cwd'); | ||||
|  | ||||
|         store.dispatch(move(active, cwd)); | ||||
|         store.dispatch(hideAll()); | ||||
|       } | ||||
|     } | ||||
|   ] | ||||
|   | ||||
| @@ -8,6 +8,8 @@ import menu from './menu'; | ||||
| import dialog from './dialog'; | ||||
| import settings from './settings'; | ||||
| import selectView from './select-view'; | ||||
| import spinner from './spinner'; | ||||
| import search from './search'; | ||||
|  | ||||
| export default function(state = new Immutable.Map(), action) { | ||||
|   console.log('action', action); | ||||
| @@ -15,6 +17,8 @@ export default function(state = new Immutable.Map(), action) { | ||||
|     lwd: lwd(state, action), // last working directory | ||||
|     cwd: cwd(state.get('cwd'), action), | ||||
|     files: files(state.get('files'), action), | ||||
|     search: search(state.get('search'), action), | ||||
|     spinner: spinner(state.get('spinner'), action), | ||||
|     selectView: selectView(state.get('selectView'), action), | ||||
|     activeFile: activeFile(state.get('activeFile'), action), | ||||
|     navigation: navigation(state.get('navigation'), action), | ||||
| @@ -26,5 +30,6 @@ export default function(state = new Immutable.Map(), action) { | ||||
|     deleteDialog: dialog(state, action, 'deleteDialog'), | ||||
|     errorDialog: dialog(state, action, 'errorDialog'), | ||||
|     createDialog: dialog(state, action, 'createDialog'), | ||||
|     searchDialog: dialog(state, action, 'searchDialog') | ||||
|   }); | ||||
| } | ||||
|   | ||||
| @@ -1,8 +1,8 @@ | ||||
| import { CHANGE_DIRECTORY, REFRESH, SETTINGS } from 'actions/types'; | ||||
| import listFiles from 'actions/list-files'; | ||||
| import { children } from 'api/files'; | ||||
| import store from 'store'; | ||||
| import { reportError } from 'utils'; | ||||
| import { listFiles } from 'actions/files-view'; | ||||
|  | ||||
| export default function(state = '', action) { | ||||
|   if (action.type === CHANGE_DIRECTORY) { | ||||
|   | ||||
| @@ -1,6 +1,6 @@ | ||||
| import { LIST_FILES, RENAME_FILE, DELETE_FILE, CREATE_FILE } from 'actions/types'; | ||||
| import { LIST_FILES, RENAME_FILE, DELETE_FILE, CREATE_FILE, MOVE_FILE, COPY_FILE, SEARCH } from 'actions/types'; | ||||
| import { refresh } from 'actions/files-view'; | ||||
| import { move, remove, sdcard, createFile, createDirectory } from 'api/files'; | ||||
| import { move, remove, sdcard, createFile, createDirectory, copy } from 'api/files'; | ||||
| import { show } from 'actions/dialog'; | ||||
| import store, { bind } from 'store'; | ||||
| import { reportError, type } from 'utils'; | ||||
| @@ -8,8 +8,8 @@ import { reportError, type } from 'utils'; | ||||
| let boundRefresh = bind(refresh()); | ||||
|  | ||||
| export default function(state = [], action) { | ||||
|   if (action.type === LIST_FILES) { | ||||
|  | ||||
|   if (action.type === LIST_FILES) { | ||||
|     let settings = store.getState().get('settings'); | ||||
|  | ||||
|     if (settings.showDirectoriesFirst) { | ||||
| @@ -22,7 +22,16 @@ export default function(state = [], action) { | ||||
|     if (!settings.showHiddenFiles) { | ||||
|       action.files = action.files.filter(file => { | ||||
|         return file.name[0] !== '.'; | ||||
|       }) | ||||
|       }); | ||||
|     } | ||||
|  | ||||
|     if (settings.filter) { | ||||
|       action.files = action.files.filter(file => { | ||||
|         if (type(file) === 'Directory') return true; | ||||
|  | ||||
|         let fileType = file.type.slice(0, file.type.indexOf('/')); | ||||
|         return fileType === settings.filter; | ||||
|       }); | ||||
|     } | ||||
|  | ||||
|     return action.files; | ||||
| @@ -36,34 +45,45 @@ export default function(state = [], action) { | ||||
|   } | ||||
|  | ||||
|   if (action.type === RENAME_FILE) { | ||||
|     let file = state[action.file]; | ||||
|     let all = Promise.all(action.file.map(file => { | ||||
|       return move(file, (file.path || '') + action.name); | ||||
|     })); | ||||
|  | ||||
|     move(file, (file.path || '') + action.name).then(boundRefresh, reportError); | ||||
|     all.then(boundRefresh, reportError); | ||||
|     return state; | ||||
|   } | ||||
|  | ||||
|   if (action.type === MOVE_FILE) { | ||||
|     let all = Promise.all(action.file.map(file => { | ||||
|       return move(file, action.newPath + '/' + file.name); | ||||
|     })); | ||||
|  | ||||
|     all.then(boundRefresh, reportError); | ||||
|     return state; | ||||
|   } | ||||
|  | ||||
|   if (action.type === COPY_FILE) { | ||||
|     let all = Promise.all(action.file.map(file => { | ||||
|       return copy(file, action.newPath + '/' + file.name); | ||||
|     })); | ||||
|  | ||||
|     all.then(boundRefresh, reportError); | ||||
|     return state; | ||||
|   } | ||||
|  | ||||
|   if (action.type === DELETE_FILE) { | ||||
|     let copy = state.slice(0); | ||||
|     let all = Promise.all(action.file.map(file => { | ||||
|       let path = ((file.path || '') + file.name).replace(/^\//, ''); | ||||
|       return remove(path, true); | ||||
|     })) | ||||
|  | ||||
|     if (action.file.length) { | ||||
|       for (let index of action.file) { | ||||
|         del(state, index); | ||||
|       } | ||||
|  | ||||
|       copy = copy.filter((a, i) => action.file.indexOf(i) === -1); | ||||
|     } else { | ||||
|       del(state, action.file); | ||||
|       copy.splice(action.file, 1); | ||||
|     } | ||||
|  | ||||
|     return copy; | ||||
|     all.then(boundRefresh, reportError); | ||||
|     return state; | ||||
|   } | ||||
|  | ||||
|   return state; | ||||
| } | ||||
|  | ||||
| function del(state, index) { | ||||
|   let file = state[index]; | ||||
|   return remove((file.path || '') + '/' + file.name).catch(reportError); | ||||
| function mov(file, newPath) { | ||||
|   return | ||||
| } | ||||
|   | ||||
							
								
								
									
										50
									
								
								src/js/reducers/search.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										50
									
								
								src/js/reducers/search.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,50 @@ | ||||
| import { SEARCH } from 'actions/types'; | ||||
| import store from 'store'; | ||||
| import { reportError } from 'utils'; | ||||
| import { listFiles } from 'actions/files-view'; | ||||
| import { children } from 'api/files'; | ||||
| import { type } from 'utils'; | ||||
|  | ||||
| export default function(state = '', action) { | ||||
|   if (action.type === SEARCH) { | ||||
|     search(action.keywords); | ||||
|  | ||||
|     return action.keywords; | ||||
|   } | ||||
|  | ||||
|   return state; | ||||
| } | ||||
|  | ||||
| function search(keywords) { | ||||
|   if (!keywords) { | ||||
|     let cwd = store.getState().get('cwd'); | ||||
|     console.log(cwd); | ||||
|     children(cwd, true).then(files => { | ||||
|       store.dispatch(listFiles(files)); | ||||
|     }, reportError); | ||||
|     return ''; | ||||
|   } | ||||
|   let keys = keywords.split(' '); | ||||
|  | ||||
|   // We don't want to show all the currently visible files from the | ||||
|   // first iteration | ||||
|   let once = true; | ||||
|   children('/', true).then(function showResults(files) { | ||||
|     if (!store.getState().get('search')) return; | ||||
|  | ||||
|     let current = once ? [] : store.getState().get('files'); | ||||
|     once = false; | ||||
|  | ||||
|     let filtered = files.filter(file => { | ||||
|       if (type(file) === 'Directory') { | ||||
|         let path = (file.path + file.name).replace(/^\//, ''); | ||||
|         children(path, true).then(showResults, reportError); | ||||
|       } | ||||
|       return keys.some(key => { | ||||
|         return file.name.indexOf(key) > -1; | ||||
|       }); | ||||
|     }); | ||||
|  | ||||
|     store.dispatch(listFiles(current.concat(filtered))); | ||||
|   }, reportError); | ||||
| } | ||||
							
								
								
									
										21
									
								
								src/js/reducers/spinner.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										21
									
								
								src/js/reducers/spinner.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,21 @@ | ||||
| import { SPINNER, CHANGE_DIRECTORY, LIST_FILES, REFRESH, DIALOG, CREATE_FILE, DELETE_FILE } from 'actions/types'; | ||||
|  | ||||
| export default function(state = false, action) { | ||||
|   if (action.type === SPINNER) { | ||||
|     return action.active === 'toggle' ? !state : action.active; | ||||
|   } | ||||
|  | ||||
|   if (action.type === DIALOG && action.id === 'errorDialog') { | ||||
|     return false; | ||||
|   } | ||||
|  | ||||
|   switch (action.type) { | ||||
|     case CHANGE_DIRECTORY: | ||||
|     case REFRESH: | ||||
|       return true; | ||||
|     case LIST_FILES: | ||||
|       return false; | ||||
|   } | ||||
|  | ||||
|   return state; | ||||
| } | ||||
| @@ -10,10 +10,11 @@ const DEFAULT = new Immutable.Map(Object.assign({ | ||||
| }, dialogs, menus)); | ||||
|  | ||||
| let store = createStore(reducers, DEFAULT); | ||||
| store.dispatch(changedir(DEFAULT.get('dir'))); | ||||
|  | ||||
| export function bind(action) { | ||||
|   return () => store.dispatch(action); | ||||
| } | ||||
|  | ||||
| export default store; | ||||
|  | ||||
| store.dispatch(changedir(DEFAULT.get('dir'))); | ||||
|   | ||||
| @@ -6,3 +6,4 @@ | ||||
| @import 'breadcrumb'; | ||||
| @import 'file-list'; | ||||
| @import 'dialog'; | ||||
| @import 'spinner'; | ||||
|   | ||||
| @@ -4,7 +4,9 @@ | ||||
|   align-items: center; | ||||
|  | ||||
|   width: 100vw; | ||||
|   height: 3.5rem; | ||||
|   height: 4.5rem; | ||||
|  | ||||
|   overflow-x: auto; | ||||
|  | ||||
|   padding: 8px; | ||||
|  | ||||
| @@ -16,6 +18,14 @@ | ||||
|  | ||||
|   border-bottom: 1px solid @dark-transparent; | ||||
|  | ||||
|   overflow-x: scroll; | ||||
|   overflow-y: hidden; | ||||
|   white-space: nowrap; | ||||
|  | ||||
|   span { | ||||
|     white-space: nowrap; | ||||
|   } | ||||
|  | ||||
|   i { | ||||
|     margin: 0 2px; | ||||
|   } | ||||
|   | ||||
| @@ -8,8 +8,7 @@ | ||||
|  | ||||
|   transform: translate(-50%, -50%); | ||||
|  | ||||
|  | ||||
|   width: 335px; | ||||
|   width: 90vw; | ||||
|   height: auto; | ||||
|  | ||||
|   padding: 1.5rem 1.7rem; | ||||
| @@ -18,9 +17,9 @@ | ||||
|  | ||||
|   .shadow-16; | ||||
|  | ||||
|   z-index: 3; | ||||
|   z-index: 5; | ||||
|  | ||||
|   transition: opacity 0.5s ease; | ||||
|   transition: opacity 0.3s ease; | ||||
|  | ||||
|   opacity: 0; | ||||
|   pointer-events: none; | ||||
|   | ||||
| @@ -14,10 +14,14 @@ | ||||
|  | ||||
|   p { | ||||
|     flex: 1 1; | ||||
|  | ||||
|     text-overflow: ellipsis; | ||||
|   } | ||||
|  | ||||
|   > span { | ||||
|     .thin-small; | ||||
|  | ||||
|     margin-left: 1rem; | ||||
|   } | ||||
|  | ||||
|   i { | ||||
|   | ||||
| @@ -1,5 +1,5 @@ | ||||
| .file-list { | ||||
|   height: ~'calc(100vh - 13.5rem)'; | ||||
|   height: ~'calc(100vh - 14.5rem)'; | ||||
|  | ||||
|   overflow-x: hidden; | ||||
|   overflow-y: auto; | ||||
|   | ||||
| @@ -15,6 +15,12 @@ header { | ||||
|  | ||||
|   h1 { | ||||
|     margin-left: -3rem; | ||||
|  | ||||
|     flex: 1; | ||||
|   } | ||||
|  | ||||
|   i { | ||||
|     margin-right: 16px; | ||||
|   } | ||||
|  | ||||
|   button { | ||||
|   | ||||
| @@ -13,7 +13,7 @@ | ||||
|  | ||||
|   opacity: 0; | ||||
|  | ||||
|   transition: opacity 0.5s ease; | ||||
|   transition: opacity 0.3s ease; | ||||
|  | ||||
|   .shadow-8; | ||||
|  | ||||
| @@ -39,5 +39,11 @@ | ||||
|     &:last-of-type { | ||||
|       border-bottom: none; | ||||
|     } | ||||
|  | ||||
|     &.disabled { | ||||
|       color: @overlay; | ||||
|  | ||||
|       pointer-events: none; | ||||
|     } | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -9,20 +9,28 @@ nav { | ||||
|   width: 70vw; | ||||
|   height: 100vh; | ||||
|  | ||||
|   overflow-y: auto; | ||||
|  | ||||
|   background: @dark; | ||||
|   color: white; | ||||
|  | ||||
|   box-shadow: 3px 0 16px 5px @dark-transparent; | ||||
|   box-shadow: 3px 0 16px 5px transparent, | ||||
|               0 0 0 1000px rgba(0, 0, 0, 0); | ||||
|   z-index: 6; | ||||
|  | ||||
|   transition: left 0.5s ease; | ||||
|   transition: left 0.3s ease, box-shadow 0.3s ease; | ||||
|  | ||||
|   ul:last-of-type li:last-of-type { | ||||
|     margin-bottom: 13px; | ||||
|   } | ||||
|  | ||||
|   &.active { | ||||
|     left: 0; | ||||
|     box-shadow: 3px 0 16px 5px @dark-transparent, | ||||
|                 0 0 0 1000px rgba(0, 0, 0, 0.55); | ||||
|  | ||||
|     i { | ||||
|       pointer-events: all; | ||||
|       opacity: 0.99; | ||||
|     } | ||||
|   } | ||||
|  | ||||
| @@ -39,6 +47,8 @@ nav { | ||||
|   } | ||||
|  | ||||
|   li { | ||||
|     display: flex; | ||||
|  | ||||
|     .light-medium; | ||||
|  | ||||
|     padding: 1rem 0 1rem 3rem; | ||||
| @@ -52,26 +62,27 @@ nav { | ||||
|       padding-bottom: 0; | ||||
|       border-bottom: none; | ||||
|     } | ||||
|  | ||||
|     label { | ||||
|       flex: 1; | ||||
|       order: 0; | ||||
|     } | ||||
|  | ||||
|     input { | ||||
|       order: 1; | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   i { | ||||
|     display: block; | ||||
|  | ||||
|     position: fixed; | ||||
|     left: 0; | ||||
|     left: 70vw; | ||||
|     top: 0; | ||||
|  | ||||
|     pointer-events: none; | ||||
|  | ||||
|     width: 100vw; | ||||
|     width: 30vw; | ||||
|     height: 100vh; | ||||
|  | ||||
|     background: rgba(0, 0, 0, 0.55); | ||||
|  | ||||
|     opacity: 0; | ||||
|  | ||||
|     z-index: -1; | ||||
|  | ||||
|     transition: opacity 0.5s ease; | ||||
|     pointer-events: none; | ||||
|   } | ||||
| } | ||||
|   | ||||
							
								
								
									
										56
									
								
								src/less/components/spinner.less
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										56
									
								
								src/less/components/spinner.less
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,56 @@ | ||||
| .sk-cube-grid { | ||||
|   opacity: 0; | ||||
|   pointer-events: none; | ||||
|  | ||||
|   width: 4rem; | ||||
|   height: 4rem; | ||||
|  | ||||
|   position: fixed; | ||||
|   left: 50%; | ||||
|   top: 50%; | ||||
|  | ||||
|   margin-top: -2rem; | ||||
|   margin-left: -2rem; | ||||
|  | ||||
|   z-index: 5; | ||||
|  | ||||
|   transition: opacity 0.3s ease; | ||||
|  | ||||
|   &.show { | ||||
|     opacity: 1; | ||||
|   } | ||||
| } | ||||
|  | ||||
| .sk-cube-grid .sk-cube { | ||||
|   width: 33.33%; | ||||
|   height: 33.33%; | ||||
|   background-color: #333; | ||||
|   float: left; | ||||
|   animation: sk-cubeGridScaleDelay 1.3s infinite ease-in-out; | ||||
| } | ||||
| .sk-cube-grid .sk-cube1 { | ||||
|   animation-delay: 0.2s; } | ||||
| .sk-cube-grid .sk-cube2 { | ||||
|   animation-delay: 0.3s; } | ||||
| .sk-cube-grid .sk-cube3 { | ||||
|   animation-delay: 0.4s; } | ||||
| .sk-cube-grid .sk-cube4 { | ||||
|   animation-delay: 0.1s; } | ||||
| .sk-cube-grid .sk-cube5 { | ||||
|   animation-delay: 0.2s; } | ||||
| .sk-cube-grid .sk-cube6 { | ||||
|   animation-delay: 0.3s; } | ||||
| .sk-cube-grid .sk-cube7 { | ||||
|   animation-delay: 0s; } | ||||
| .sk-cube-grid .sk-cube8 { | ||||
|   animation-delay: 0.1s; } | ||||
| .sk-cube-grid .sk-cube9 { | ||||
|   animation-delay: 0.2s; } | ||||
|  | ||||
| @keyframes sk-cubeGridScaleDelay { | ||||
|   0%, 70%, 100% { | ||||
|     transform: scale3D(1, 1, 1); | ||||
|   } 35% { | ||||
|     transform: scale3D(0, 0, 1); | ||||
|   } | ||||
| } | ||||
| @@ -50,3 +50,18 @@ | ||||
|   width: 6px; | ||||
|   height: 24px; | ||||
| } | ||||
|  | ||||
| .icon-search { | ||||
|   .icon; | ||||
|   background: url(/img/Search.svg) no-repeat; | ||||
|   width: 19px; | ||||
|   height: 27px; | ||||
| } | ||||
|  | ||||
| .icon-cross { | ||||
|   .icon-plus; | ||||
|   transform: rotate(45deg); | ||||
|   svg * { | ||||
|     fill: white; | ||||
|   } | ||||
| } | ||||
|   | ||||
| @@ -25,3 +25,9 @@ body { | ||||
|  | ||||
|   flex-flow: column; | ||||
| } | ||||
|  | ||||
| a { | ||||
|   color: currentColor; | ||||
|  | ||||
|   text-decoration: none; | ||||
| } | ||||
|   | ||||
| @@ -2,3 +2,37 @@ | ||||
| @import 'shadows'; | ||||
| @import 'buttons'; | ||||
| @import 'forms'; | ||||
|  | ||||
| .coming-soon::after { | ||||
|   content: 'soon...'; | ||||
|  | ||||
|   background: @cream; | ||||
|  | ||||
|   color: @dark; | ||||
|  | ||||
|   padding: 2px 8px; | ||||
|  | ||||
|   border-radius: 12px; | ||||
|  | ||||
|   font-size: 11px; | ||||
|   font-weight: normal; | ||||
| } | ||||
|  | ||||
| li.coming-soon::after { | ||||
|   margin-right: 13px; | ||||
|   float: right; | ||||
| } | ||||
|  | ||||
| button.coming-soon { | ||||
|   position: relative; | ||||
| } | ||||
|  | ||||
| button.coming-soon::after { | ||||
|   position: absolute; | ||||
|   left: 50%; | ||||
|   top: 50%; | ||||
|  | ||||
|   opacity: 0.8; | ||||
|  | ||||
|   transform: translate(-50%, -50%) rotate(-45deg); | ||||
| } | ||||
|   | ||||
| @@ -14,9 +14,8 @@ input { | ||||
|   .light-medium; | ||||
| } | ||||
|  | ||||
| label { | ||||
|   clear: left; | ||||
|  | ||||
| input[type='checkbox'] + label, | ||||
| input[type='radio'] + label { | ||||
|   &::after { | ||||
|     content: ''; | ||||
|     display: block; | ||||
| @@ -36,10 +35,8 @@ label { | ||||
|   } | ||||
| } | ||||
|  | ||||
| input[type='checkbox'] { | ||||
|   clear: right; | ||||
|   float: right; | ||||
|  | ||||
| input[type='checkbox'], | ||||
| input[type='radio'] { | ||||
|   display: none; | ||||
| } | ||||
|  | ||||
|   | ||||
| @@ -5,6 +5,7 @@ | ||||
| @gray: #F0F0F0; | ||||
| @background: #FAFAFA; | ||||
| @blue: #63B0CD; | ||||
| @cream: #F7C59F; | ||||
|  | ||||
| @success: #B8E986; | ||||
|  | ||||
|   | ||||
		Reference in New Issue
	
	Block a user