bisgit 0.6.0 → 0.8.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +4 -4
- package/dist/chunks/chunk-7XT4D6II.js +2 -0
- package/dist/chunks/{chunk-L4626BRA.js → chunk-HKY7QTE6.js} +1 -1
- package/dist/chunks/commands-6NWVWJQB.js +13 -0
- package/dist/chunks/pages-YLGVYIE5.js +2 -0
- package/dist/main.js +1 -1
- package/package.json +7 -7
- package/dist/chunks/chunk-DLHREY2Q.js +0 -2
- package/dist/chunks/commands-XBKP3MSC.js +0 -12
- package/dist/chunks/pages-JELX53G4.js +0 -2
package/README.md
CHANGED
|
@@ -2,7 +2,7 @@
|
|
|
2
2
|
|
|
3
3
|
<img alt="biscuit with jelly" src="https://github.com/alexanderdombroski/bisgit/blob/main/public/bisgit.png" width="200px" />
|
|
4
4
|
|
|
5
|
-
|
|
5
|
+
Run `gi` or `bisgit` for full TUI.
|
|
6
6
|
|
|
7
7
|
## Commands
|
|
8
8
|
|
|
@@ -18,7 +18,8 @@ Full Tui coming soon
|
|
|
18
18
|
- `gi backmerge <branch>` updates a branch and then merges it into the current branch.
|
|
19
19
|
- `gi conflict <branch>` shows all conflicts that would occur in a merge.
|
|
20
20
|
- `gi abort` and `gi continue` are useful in conflict resolution situations.
|
|
21
|
-
|
|
21
|
+
- `gi files <ref>` shows files of the commit or stash.
|
|
22
|
+
- `gi lines <ref> <ref?>` shows the number of lines changed. Use `-v` for verbose output.
|
|
22
23
|
|
|
23
24
|
### Branching
|
|
24
25
|
|
|
@@ -32,13 +33,12 @@ Full Tui coming soon
|
|
|
32
33
|
- `gi code-review <pr>` checks out a pr and creates a diff similar to github's code review diff viewer.
|
|
33
34
|
- `gi whoami` shows github username.
|
|
34
35
|
- `gi languages` prints the percentages of languages for this repo.
|
|
35
|
-
- `gi coauthor <username
|
|
36
|
+
- `gi coauthor <username?>` commits with a co-authorship description.
|
|
36
37
|
|
|
37
38
|
### Utility
|
|
38
39
|
|
|
39
40
|
- `gi pwd` shows repo root path.
|
|
40
41
|
- `gi wipe` clears all uncommitted trackable files
|
|
41
|
-
- `gi files <commit>` shows the files edited by the given commit.
|
|
42
42
|
- `gi churn` shows you the 25 most edited files.
|
|
43
43
|
- `gi exclude` and `gi include` allows you to ignore files locally without modifying the .gitignore file.
|
|
44
44
|
- `gi remote-default` shows whether the remote default branch is 'main', 'master', etc.
|
|
@@ -0,0 +1,2 @@
|
|
|
1
|
+
#!/usr/bin/env node
|
|
2
|
+
import{b as n,c,d as m}from"./chunk-R7DNYVL6.js";import h from"fs/promises";async function T(t){try{return await h.access(t),!0}catch{return!1}}async function p(){return await c("git ls-files")}async function y(){return await c("git ls-files --others --exclude-standard")}async function v(){let[t,e]=await Promise.all([p(),y()]),r=[...t,...e].sort(([s],[o])=>s.localeCompare(o)),i={};for(let s of r){let o=s.split("/"),a=i;for(let u of o)a[u]||(a[u]={}),a=a[u]}return i}function w(t,e=""){let r=[];for(let[i,s]of Object.entries(t)){let o=e?`${e}/${i}`:i;r.push(o),Object.keys(s).length>0&&r.push(...w(s,o))}return r}import{normalize as f}from"node:path";async function x(){let{stdout:t}=await n("git branch --show-current");return t.trim()}async function C(t){t??=await x();let{stdout:e}=await n(`git config --get branch.${t}.remote`);return e.trim()}async function N(t,e){try{let{stdout:r}=await n(`git rev-list --count ${t}/${e}..${e}`);return parseInt(r.trim(),10)}catch(r){return console.error("Error getting commits ahead:",r),NaN}}async function E(t,e){try{let{stdout:r}=await n(`git rev-list --count ${e}..${t}/${e}`);return parseInt(r.trim(),10)}catch(r){return console.error("Error getting commits behind:",r),NaN}}async function F(t){let{stdout:e}=await n(`git rev-parse --git-path ${t}`);return f(e.trim())}async function S(){let{stdout:t}=await n("git rev-parse --git-dir");return f(t.trim())}async function k(){let{stdout:t}=await n("git rev-parse --show-toplevel");return f(t.trim())}async function G(t){let{code:e}=await m("git",["remote","get-url",t],{stdio:"ignore"});return e===0}async function I(){let{stdout:t}=await n("git rev-parse --abbrev-ref HEAD");return t.trim()==="HEAD"}async function M(t,e){let r="git fetch";t&&(r+=` ${t}`),e&&(r+=` ${e}`),await n(r)}async function z(){let{stdout:t}=await m("git",["status","--porcelain"]);return!t?.trim()}async function A(){return await c("git status --porcelain")}async function $(t,e){let{stdout:r}=await n(`git merge-base ${t} ${e}`);return r.trim()}async function H(t,e){let r=await $(t,e),{stdout:i}=await m("git",["merge-tree",r,e,t]);return!i?.split(/\r?\n/)?.includes("+=======")}async function O(t){let{stdout:e}=await n(`git rev-parse ${t}`);return e.trim()}async function j(t,e,r){return await c(`git rev-list ${r?"--reverse":""} ${t}..${e}`)}async function V(){return await c("git for-each-ref --format='%(refname:short)' refs/heads")}async function q(){let{stdout:t}=await n("git remote -v");return t.trim().split(/\r?\n/).map(r=>{let i=r.match(/^(\S+)\s+(\S+)\s+\((fetch|push)\)$/);if(!i)return null;let[,s,o,a]=i;return{name:s,url:o,type:a}}).filter(r=>r!==null)}async function J(t){try{let{stdout:e}=await n(`git log -1 --format='%an|%ae|%ad|%h' -- "${t}"`,{encoding:"utf8"}),[r,i,s,o]=e.trim().split("|");return{authorName:r,authorEmail:i,authorDate:s,shortHash:o}}catch{return{}}}async function K(t,e){let r=e?`git diff --numstat ${t} ${e}`:b(t)?`git stash show --numstat --format="" --include-untracked ${t}`:`git show --numstat --format="" ${t}`;return(await c(r)).map(s=>{let[o,a,u]=s.split(" ");if(o==="-"&&a==="-")return{file:u,added:null,deleted:null,changeType:"binary"};let g=Number(o),l=Number(a),d="modified";return g>0&&l===0?d="added":g===0&&l>0&&(d="deleted"),{file:u,added:g,deleted:l,changeType:d}})}function b(t){return/^stash@\{\d+\}$/.test(t)}export{T as a,v as b,w as c,x as d,C as e,N as f,E as g,F as h,S as i,k as j,G as k,I as l,M as m,z as n,A as o,$ as p,H as q,O as r,j as s,V as t,q as u,J as v,K as w,b as x};
|
|
@@ -1,2 +1,2 @@
|
|
|
1
1
|
#!/usr/bin/env node
|
|
2
|
-
import{a as s,l as i,n}from"./chunk-
|
|
2
|
+
import{a as s,l as i,n}from"./chunk-7XT4D6II.js";function u(){return process.argv[2]}function g(){return process.argv.slice(3)}async function p(){let[r,t]=g();return r||(console.error("Requires <branch> or <remote> <branch>"),process.exit(1)),t?{remote:r,branch:t}:{remote:"origin",branch:r}}import{spawnSync as a}from"node:child_process";function e(r,t,o){let{status:c}=a(r,t,{stdio:"ignore"});c!==0&&(o&&console.error(o),process.exit(1))}var h=()=>e("git",["rev-parse","--git-dir"],"Need to use command in a git repository."),x=()=>e("git",["rev-parse","HEAD"],"Need to create first commit."),q=(r="origin")=>e("git",["remote","get-url",r],`Need to add remote '${r}'.`),v=r=>e("git",["show-ref","--branches",r],`Branch '${r}' isn't tracked locally and may not exist.`),y=async()=>{await n()||(console.error("You should stash or commit your changes first."),process.exit(1))},b=async r=>{await s(r)||(console.error(`${r} must exist for this operation and it doesn't for some reason`),process.exit(1))},w=async()=>{await i()&&(console.error("Operation not allowed in a 'detached HEAD' state"),process.exit(1))},A=r=>e("git",["rev-parse","--abbrev-ref","--symbolic-full-name",`${r}@{u}`],`branch '${r}' has no upstream`);function C(r="Error: missing required argument"){if(process.argv[3])return process.argv[3];console.error(r),process.exit(1)}async function $(){let{status:r}=a("git",["diff","--staged","--quiet"],{stdio:"ignore"});r===0&&(console.error("You have no staged changes"),process.exit(1))}export{u as a,g as b,p as c,h as d,x as e,q as f,v as g,y as h,b as i,w as j,A as k,C as l,$ as m};
|
|
@@ -0,0 +1,13 @@
|
|
|
1
|
+
#!/usr/bin/env node
|
|
2
|
+
import{b as g,c as X,d as at,e as R,f as w,g as V,h as A,i as N,j as U,k as ct,l as y,m as S}from"./chunk-HKY7QTE6.js";import{a as C,d as h,e as tt,f as W,g as G,h as k,i as et,j as ot,k as rt,m as I,n as x,o as M,p as P,q as it,r as nt,s as F,t as st,w as H,x as L}from"./chunk-7XT4D6II.js";import{a as b,b as s,c as E,d,e as f}from"./chunk-R7DNYVL6.js";import B from"node:path";async function mt(){let t=await M(),e=/^(UU|DU|UD)/;return!!t?.some(o=>e.test(o))}async function j(){let t=await et(),[e,o,r,i,n]=await Promise.all([await C(B.join(t,"MERGE_HEAD")),await C(B.join(t,"rebase-merge")),await C(B.join(t,"rebase-apply")),await C(B.join(t,"CHERRY_PICK_HEAD")),await C(B.join(t,"REVERT_HEAD"))]);return{merge:e,rebaseMerge:o,rebaseApply:r,cherryPick:i,revert:n}}async function lt(){let{merge:t,rebaseMerge:e,rebaseApply:o,cherryPick:r,revert:i}=await j(),n="";if(t)n="merge";else if(e||o)n="rebase";else if(r)n="cherry-pick";else if(i)n="revert";else{let a=await mt()?"Conflicts exist, but likely are from a stash and are not abortable. Resolve them manually.":"Nothing to abort.";console.log(a);return}console.log(`Aborting the ${n}`),await f([n,"--abort"])}async function ft(){await U(),await S(),b("git",["commit","--amend","--no-edit",...g()],{stdio:"inherit",silent:!0,triggerExit:!0})}import{render as re}from"ink";import{render as ee}from"ink";import{Suspense as zt,use as Jt,useEffect as Kt}from"react";import{Text as pt,useApp as Zt}from"ink";import Qt from"ink-spinner";import{jsx as O,jsxs as te}from"react/jsx-runtime";function m(t){let{msg:e}=t;return O(zt,{fallback:te(pt,{children:[O(Qt,{type:"dots"}),"\xA0",e]}),children:O(Xt,{...t})})}function Xt({msg:t,promise:e}){let o=Jt(e),{exit:r}=Zt();return Kt(()=>{r()},[o]),O(pt,{children:`\u2714 ${typeof o=="string"?o:t}`})}import{jsx as oe}from"react/jsx-runtime";async function gt(){let t=process.argv[3],e=process.argv[4]??await h();if(V(t),V(e),t===e)return console.log(`No reason to merge '${t}' into '${e}'`);let o=await tt(t);w(o),ee(oe(m,{msg:"Updating target branch and merging",promise:(async()=>{try{await T(o,t)}catch{console.error(`Error when updating branch ${t} with remote ${o}. Fix divergent branches before merging into ${e}`);return}await h()!==e&&await s(`git switch ${e}`),b("git",["merge",t],{stdio:"inherit",silent:!0})})()}))}async function T(t,e){await I(t,e);let[o,r,i]=await Promise.all([W(t,e),G(t,e),h()]);if(r!==0){if(o===0&&i!==e){await s(`git fetch ${t} ${e}:${e}`);return}if(await s(`git switch ${e}`),o===0){await s(`git merge ${t}/${e}`);return}await f(["merge",`${t}/${e}`],["ignore","ignore","inherit"])}}async function q(){R(),w();let{stdout:t,code:e}=await d("git",["rev-parse","--abbrev-ref","origin/HEAD"]);if(e===0)return ut(t.trim());await s("git remote set-head origin -a");let{stdout:o}=await s("git rev-parse --abbrev-ref origin/HEAD");return ut(o.trim())}async function dt(){let{remote:t,branch:e}=await q();console.log(`Remote default branch is '${t}/${e}'`)}async function ut(t){let e=t.split("/"),o=t.split("/").map((r,i)=>e.slice(0,i+1).join("/"));for(let r of o)if(await rt(r)){let i=t.slice(r.length+1);return{remote:r,branch:i}}throw new Error("Failed to get default remote")}import{jsx as ne}from"react/jsx-runtime";var ie=["master","main","development","lingoport"];async function ht(){await A();let{branch:t,remote:e}=await q();await s(`git switch ${t}`);let o=(async()=>{await f(["fetch","--prune","--prune-tags"]),await T(e,t)})();re(ne(m,{msg:"Fetching and pruning remote references",promise:o})),await o;let r=await st();for(let i of r){if(i===t||ie.includes(i))continue;let{code:n}=await d("git",["diff","--quiet",t,i]);if(n===0){Y(i);continue}let{code:a}=await d("git",["merge-base","--is-ancestor",i,t]);if(a===0){Y(i);continue}let c=await P(i,t),{stdout:l}=await d("git",["merge-tree",c,t,i]);l?.trim()||Y(i)}}function Y(t){f(["branch","-D",t])}import{spawn as se}from"node:child_process";import{createInterface as ae}from"node:readline/promises";async function wt(){R;let e=[...(await ce()).entries()].sort(([,r],[,i])=>i-r).slice(0,25),o=e[0][1].toString().length;console.info(e.map(([r,i])=>`${String(i).padEnd(o)} ${r}:`).join(`
|
|
3
|
+
`))}async function ce(){let{promise:t,resolve:e,reject:o}=Promise.withResolvers(),r=new Map,i=se("git",["log","--all","-M","-C","--name-only","--format=format:"]),n=ae({input:i.stdout,crlfDelay:1/0});return n.on("line",a=>{let c=a.trim();c&&r.set(c,(r.get(c)??0)+1)}),n.on("close",()=>e(r)),i.on("error",o),t}import{render as bt}from"ink";import{Box as me,Text as le,useApp as fe,useInput as pe}from"ink";import ge from"ink-select-input";import{jsx as yt,jsxs as ue}from"react/jsx-runtime";function $t({prompt:t,onSelect:e,items:o}){let{exit:r}=fe();return pe((i,n)=>{n.escape&&r()}),ue(me,{flexDirection:"column",children:[yt(le,{children:t}),yt(ge,{items:o,onSelect:e})]})}import{jsx as xt}from"react/jsx-runtime";async function Pt(){await S();let t=g()[0];if(!t){let n=(await he()).map(c=>({value:c,label:`${c.name} ${c.email} (${c.commits})`}));bt(xt($t,{items:n,prompt:"Who assisted in this commit?",onSelect:({value:{name:c,email:l}})=>{Ct(c,l.slice(1,-1))}}));return}let e=(async()=>{let i=await de(t);return i.email||(console.error(`Couldn't find an email for ${t}`),process.exit(1)),i})();bt(xt(m,{msg:"Fetching name and email",promise:e}));let{name:o,email:r}=await e;Ct(o,r)}async function de(t){let{stdout:e}=await s(`gh api users/${t}`);return JSON.parse(e.trim())}async function he(){let t=await E('git log --format="%an|<%ae>"'),e=new Map;for(let o of t){let[r,i]=o.split("|"),n=`${r}|${i}`;e.has(n)||e.set(n,{name:r,email:i,commits:0}),e.get(n).commits+=1}return[...e.values()].toSorted((o,r)=>r.commits-o.commits)}async function Ct(t,e){f(["commit","--edit","-m","<insert summary>","-m",`Co-authored-by: ${t} <${e}>`])}import{render as z}from"ink";import{isDeepStrictEqual as we}from"node:util";import{jsx as J}from"react/jsx-runtime";async function Rt(){let t=y("Error: missing id of PR to checkout");await A();let e=(async()=>Promise.all([s(`gh co ${t}`),s(`gh pr view ${t} --json baseRefName -q .baseRefName`)]))();z(J(m,{msg:`Checking out pr #${t}`,promise:e}));let[,{stdout:o}]=await e,r=o.trim(),i=(async()=>{await s(`git switch ${r}`),await s("git pull"),await s("git switch -")})();z(J(m,{msg:`Updating ${r} branch merge destination`,promise:i})),await i;let n=(async()=>{let p=await P(r,"HEAD");return await s(`git reset ${p} --soft`),await ye("HEAD")})();z(J(m,{msg:"Creating an editable, explorable, local PR experience",promise:n}));let[a,c]=await Promise.all([n,$e(t)]),{files:l,additions:u,deletions:D}=c;if(we(a,c))return console.info(`Files: ${l}, +${u} -${D}`);console.info("The PR stat doesn't match the local stat."),console.info("LOCAL"),console.table(a),console.info("GITHUB"),console.table(c),console.info("One option to fix (there'll likely be conflicts):"),console.info("1. git stash -u"),console.info(`2. git merge ${r}`),console.info("3. git stash pop")}async function ye(t){let{stdout:e}=await s(`git diff --shortstat ${t}`),o=/(\d+)\s+files? changed(?:, (\d+) insertions?\(\+\))?(?:, (\d+) deletions?\(-\))?/,r=e.trim().match(o);if(!r)throw new Error(`Couldn't generate short stat for ref(s): ${t}`);let[,i,n,a]=r;return{files:parseInt(i,10),additions:n?parseInt(n,10):0,deletions:a?parseInt(a,10):0}}async function $e(t){let{stdout:e}=await s(`gh pr view ${t} --json deletions,files,additions`),o=JSON.parse(e.trim());return o.files=o.files.length,o}import{spawn as be}from"node:child_process";import{createInterface as Ce}from"node:readline/promises";import K from"chalk";async function At(){let t=y("Error: missing target branch argument"),e=await P("HEAD",t),{promise:o,resolve:r,reject:i}=Promise.withResolvers(),n=be("git",["merge-tree",e,"HEAD",t]),a=Ce({input:n.stdout,crlfDelay:1/0}),c=!1,l=0;a.on("line",p=>{p=p.trimEnd(),p==="+<<<<<<< .our"?(c=!0,console.info(K.red(p)),l+=1):p==="+>>>>>>> .their"?(c=!1,console.info(K.green(p))):p==="+======="?console.info(K.yellow(p)):c&&console.info(p)}),a.on("close",()=>r(l)),n.on("error",i);let u=await o,D=u?`\u274C ${u} conflicts would occur!`:"\u2705 No conflicts. Safe to merge.";console.info(D)}async function St(){let{merge:t,rebaseMerge:e,rebaseApply:o,cherryPick:r,revert:i}=await j(),n="";if(t)n="merge";else if(e||o)n="rebase";else if(r)n="cherry-pick";else if(i)n="revert";else{console.log("Nothing to continue.");return}console.log(`Continuing the ${n}`),await f([n,"--continue"])}import{appendFileSync as Pe}from"node:fs";import{readFile as xe}from"node:fs/promises";async function _(t){return(await xe(t,"utf-8")).split(/\r?\n/).map(o=>o.trim()).filter(o=>o&&!o.startsWith("#"))}async function vt(){let t=await k("info/exclude");await N(t);let e=await _(t);g().forEach(o=>{e.includes(o)?console.info(`Already Exists: '${o}'`):(Pe(t,`${o}
|
|
4
|
+
`),console.info(`Added '${o}'`))}),console.info(`
|
|
5
|
+
See file at ${t}`)}async function Dt(){let t=g();if(t[0]){let e=await Re(t[0]);console.info(e.join(`
|
|
6
|
+
`))}else f(["status","--short"])}async function Re(t){return L(t)?await E(`git stash show -p --include-untracked --name-only ${t}`):await E(`git show --name-only --pretty=format:"" ${t}`)}import{spawn as Ae}from"node:child_process";async function Et(){let t=y("Error: missing valid target ref argument");await S();let e=await nt(t);await f(["commit",`--fixup=${e}`]);let r=(await F(`${e}^`,"HEAD",!0)).map(n=>Se(n));(await Promise.all(r)).every(Boolean)||(console.warn("Rebase conflict would occur!"),await s("git reset HEAD~1 --soft"),process.exit(1)),await s(`git rebase -i --autosquash ${e}^`)}async function Se(t){let{stdout:e}=await s(`git diff ${t}^ ${t}`),{promise:o,resolve:r,reject:i}=Promise.withResolvers(),n=Ae("git",["apply","--check","--3way","-q"],{stdio:["pipe","ignore","ignore"]});return n.on("close",a=>{r(a===0)}),n.on("error",i),n.stdin.write(e),n.stdin.end(),o}import{readFile as ve,writeFile as De}from"node:fs/promises";async function kt(){let t=await k("info/exclude");await N(t);let e=await _(t),o=g();o.filter(a=>!e.includes(a)).forEach(a=>console.info(`No match: '${a}'`));let i=(await ve(t,"utf-8")).split(/\r?\n/).map(a=>a.trim()),n=i.filter(a=>!o.includes(a));await De(t,n.join(`
|
|
7
|
+
`)),console.info(`
|
|
8
|
+
Removed ${i.length-n.length} entries from ${t}`)}import{render as Ee}from"ink";import{jsx as qe}from"react/jsx-runtime";async function Bt(){let t=(async()=>{let i=await ke();return await Be(i)})();Ee(qe(m,{msg:"Fetching repo languages",promise:t}));let e=await t,o=Object.values(e).reduce((i,n)=>i+n,0),r=Object.entries(e).map(([i,n])=>`${i}: ${Math.round(n/o*100)}% - ${Te(n)}`);console.log(r.join(`
|
|
9
|
+
`))}async function ke(){let{stdout:t}=await s("gh repo view --json nameWithOwner -q .nameWithOwner");return t.trim()}async function Be(t){let{stdout:e}=await s(`gh api repos/${t}/languages`);return JSON.parse(e.trim())}function Te(t,e=2){if(t===0)return"0 B";let o=1024,r=e<0?0:e,i=["B","KB","MB","GB","TB","PB","EB","ZB","YB"],n=Math.floor(Math.log(t)/Math.log(o));return`${parseFloat((t/Math.pow(o,n)).toFixed(r))} ${i[n]}`}import $ from"chalk";async function Tt(){y();let t=g(),[e,o]=t.filter(i=>i!=="-v"),r=[];if(L(e)?(r.push(...await H(`${e}^3`)),r.push(...await H(`${e}^2`))):r.push(...await H(e,o)),t.includes("-v")){let i={added:"A",modified:"M",deleted:"D",binary:"B"};r.forEach(n=>{let c=`${We(n.changeType)(i[n.changeType])} ${n.file} |`;n.added&&(c+=$.green(` +${n.added}`)),n.deleted&&(c+=$.red(` -${n.deleted}`)),console.info(c)})}else{let i=new Set(r.map(({file:u})=>u)).size,[n,a]=r.reduce((u,{added:D,deleted:p})=>[u[0]+(D??0),u[1]+(p??0)],[0,0]),c=$.green(`+${n}`),l=$.red(`-${a}`);console.info(`${i} files: ${c} ${l}`)}}var We=t=>{switch(t){case"modified":return $.yellow;case"added":return $.green;case"deleted":return $.red;default:return e=>e}};async function qt(){let t=await ot();console.log(t)}import{render as Gt}from"ink";function Wt(){return new Date().toLocaleString("en-US",{year:"2-digit",month:"2-digit",day:"2-digit",hour:"numeric",minute:"2-digit",hour12:!0}).replace(",","")}async function v(){let t=Wt();await s("git add -A"),await f(["commit","-m",`"WIP ${t}"`])}import{nanoid as Ge}from"nanoid";import Ie from"chalk";import{jsx as It}from"react/jsx-runtime";async function Mt(){let{remote:t,branch:e}=await q(),o=process.argv[3]??e;await x()||await v(),await it(o,"HEAD")||(console.warn("Conflicts would occur. Not safe to rebranch."),process.exit(1));let r=await h(),i=(async()=>{await T(t,o)})();Gt(It(m,{msg:`Updating branch ${o}`,promise:i})),await i;let n=`${Ge(8)}`,a=(async()=>{await s(`git switch ${r}`),await s(`git branch ${n}`),await s(`git switch ${o}`),await s(`git branch -D ${r}`),await s(`git switch -c ${r}`);let c=await F(o,n,!0);for(let l of c)try{await s(`git cherry-pick ${l}`)}catch{let u=Ie.yellow(`git branch -D ${r} && git branch -m ${n} ${r}`);throw new Error(`\u274C cherry pick failed at ${l}
|
|
10
|
+
Run this to restore backup branch
|
|
11
|
+
> ${u}`)}await s(`git branch -D ${n}`)})();Gt(It(m,{msg:`Rewriting branch ${r}`,promise:a})),await a,console.info(`Successfully recreated branch ${r} from ${o}`)}import{execSync as Me}from"node:child_process";async function Ft(){R();let t=Fe(process.argv[3]),{default:e}=await import("clipboardy");await e.write(t),console.log(`'${t}' copied to clipboard`)}function Fe(t="HEAD"){return Me(`git rev-parse --short ${t}`,{encoding:"utf-8"}).trim()}import{render as He}from"ink";import{jsx as Le}from"react/jsx-runtime";async function Ht(){let{remote:t,branch:e}=await X();w(t),await A();let o=I(t,e);He(Le(m,{msg:`Fetching branch ${e} from ${t}`,promise:o})),await o,await f(["switch","-c",e,"--track",`${t}/${e}`])}async function Z(){let{stdout:t}=await s("npm list -g bisgit --depth=0 --json");return JSON.parse(t.trim()).dependencies.bisgit.version}async function Lt(){console.info(await Z())}async function Ne(){let{code:t}=await d("npm",["outdated","-g","bisgit"]);return t===0}async function Ue(){let{stdout:t}=await s("npm view bisgit version",{encoding:"utf8"});return t.trim()}async function Nt(){if(await Ne())return console.info("Already up to date");console.info(`Updating: ${await Z()} -> ${await Ue()}`),b("npm",["i","-g","bisgit@latest"])}async function Ut(){let[t,e]=await Promise.allSettled([await Oe("user.name"),await je()]);t.status==="fulfilled"&&console.info(`Git: ${t.value}`),e.status==="fulfilled"&&console.info(`Github: ${e.value}`)}async function je(){let{stdout:t}=await s("gh api user --jq .login");return t.trim()}async function Oe(t){let{stdout:e}=await s(`git config ${t}`);return e.trim()}import{render as Ze}from"ink";import{Text as _e,useApp as Ve,useInput as Ye}from"ink";import{useState as ze}from"react";import{Fragment as Ke,jsx as Je,jsxs as jt}from"react/jsx-runtime";function Ot({prompt:t,msg:e,onConfirm:o}){let[r,i]=ze(""),{exit:n}=Ve();return Ye((a,c)=>{if(r)return;let l=a.toLowerCase();l==="y"?i("Yes"):(l==="n"||c.return)&&(i("No"),n())}),jt(Ke,{children:[jt(_e,{children:[t," [y/N] ",r]}),r==="Yes"&&Je(m,{msg:e,promise:o()})]})}import{jsx as Qe}from"react/jsx-runtime";async function _t(){if(at(),await x())return console.log("Already a clean working tree");let t=await M();if(!t)throw new Error("git status cmd failed");let e=new Q(t);e.displayReport(),Ze(Qe(Ot,{prompt:"Approve these changes?",msg:"Running git reset && git clean",onConfirm:async()=>(await s("git reset --hard"),await s("git clean -f"),`Altered ${e.willDelete.length+e.willRecreate.length+e.willRevert.length} files.`)}))}var Q=class{willDelete=[];willRecreate=[];willRevert=[];constructor(e){e.forEach(o=>{if(o){let r=o.slice(0,2).trim(),i=o.slice(3);r.includes("?")||r.includes("A")?this.willDelete.push(i):r==="DU"||r==="D"?this.willRecreate.push(i):this.willRevert.push(i)}}),this.willDelete.sort(),this.willRecreate.sort(),this.willRevert.sort()}displayReport(){this.displaySection("Will Recreate:",this.willRecreate),this.displaySection("Will Revert:",this.willRevert),this.displaySection("Will Delete:",this.willDelete)}displaySection(e,o){o.length&&console.log(`${e}
|
|
12
|
+
${o.map(r=>`- ${r}`).join(`
|
|
13
|
+
`)}`)}};import{nanoid as Xe}from"nanoid";import{render as to}from"ink";import{jsx as eo}from"react/jsx-runtime";async function Vt(){await U();let t=process.argv[3]??"origin";w(t);let e=await h();ct(e);let o="",i=(async()=>{let n=[s(`git fetch ${t} ${e} --force`)];await x()||n.push(v()),await Promise.all(n);let[a,c]=await Promise.all([W(t,e),G(t,e)]);if(!(a===0&&c===0)){if(a!==0){let l=`${e}-${Xe(8)}`;o=`Created branch ${l}`,await s(`git branch ${l}`)}await s(`git reset --hard ${t}/${e}`)}})();to(eo(m,{msg:`Force pull reseting ${e} -> ${t}`,promise:i})),await i,o&&console.info(o)}var Yt={abort:lt,amend:ft,autoprune:ht,backmerge:gt,churn:wt,coauthor:Pt,"code-review":Rt,conflict:At,continue:St,exclude:vt,files:Dt,fixup:Et,include:kt,languages:Bt,lines:Tt,pwd:qt,rebranch:Mt,"remote-default":dt,savepoint:v,sha:Ft,track:Ht,update:Nt,"--version":Lt,whoami:Ut,wipe:_t,yank:Vt};async function vn(t){return await Yt[t]?.(),Object.hasOwn(Yt,t)}export{vn as runCommand};
|
|
@@ -0,0 +1,2 @@
|
|
|
1
|
+
#!/usr/bin/env node
|
|
2
|
+
import{a as Le,b as Re,c as Ne,d as Fe,j as We,o as Ie,r as Oe,t as Ke,u as $e,v as Ge,w as Q}from"./chunk-7XT4D6II.js";import{b as P,c as De}from"./chunk-R7DNYVL6.js";import{render as gn}from"ink";import{createContext as ro,useContext as no,useState as io}from"react";import{jsx as so}from"react/jsx-runtime";var He=ro(void 0),Ve=({children:e})=>{let[t,r]=io({}),o=(i,s)=>{r(a=>({...a,[i]:s}))},n=i=>{let s=Array.isArray(i)?i:[i];r(a=>{let l={...a};for(let c of s)delete l[c];return l})};return so(He.Provider,{value:{keybindings:t,setKeybinding:o,removeKeybinding:n},children:e})},M=()=>{let e=no(He);if(!e)throw new Error("useKeybindings must be used within a KeybindingsProvider");return e};import{useEffect as mn}from"react";import{Text as un,Box as to,useInput as pn}from"ink";import{useStdout as ao}from"ink";function g(){let{stdout:e}=ao(),t=e.rows-2,r=Math.floor(t/2),o=e.columns>100?96:e.columns-4;return{width:e.columns,height:e.rows,sectionHeight:t,sectionHalfHeight:r,modalWidth:o}}import{Box as Ye,Text as uo}from"ink";import{useState as se,createContext as co,useContext as lo}from"react";import{jsx as mo}from"react/jsx-runtime";var I={Files:"Files","File Meta":"Files",Status:"Status",Log:"Log","Commit Details":"Log",Branches:"Branches",Worktrees:"Branches",Remotes:"Branches",Stashes:"Stashes","Stash Details":"Stashes"},U=Object.keys(I),_e=co({});function Ue({children:e}){let[t,r]=se(!1),o=()=>r(!0),n=()=>r(!1),[i,s]=se("Status"),[a,l]=se("Status"),c=p=>{t||(l(p),s(I[p]))},m=p=>{t||(s(p),l(p))},f=()=>{t||l(p=>{let S=U.at((U.indexOf(p)+1)%U.length);return s(I[S]),S})},b=()=>{t||l(p=>{let S=U.at(U.indexOf(p)-1);return s(I[S]),S})};return mo(_e.Provider,{value:{activeGroup:i,setActiveGroup:m,activeSection:a,setActiveSection:c,prevSection:b,nextSection:f,isLocked:t,lock:o,unlock:n},children:e})}function w(){let e=lo(_e);if(Object.keys(e).length===0)throw new Error("useNavContext must be used within a NavProvider");return e}import{memo as po}from"react";import{jsx as ae}from"react/jsx-runtime";var fo=Array.from(new Set(Object.values(I))),qe=po(({activeGroup:e})=>ae(Ye,{width:"100%",flexDirection:"row",height:1,children:fo.map((t,r)=>ae(Ye,{borderStyle:"classic",borderLeft:!!r,borderRight:!1,borderTop:!1,borderBottom:!1,children:ae(uo,{color:t===e?"cyan":"",children:` [${r+1}]: ${t} `})},t))}),(e,t)=>e.activeGroup===t.activeGroup);import{Text as an,useInput as cn}from"ink";import{Box as So,Text as wo}from"ink";import{Box as Qe,measureElement as go,Text as ze}from"ink";import{useMemo as ho,useRef as xo}from"react";import{Fragment as vo,jsx as z,jsxs as yo}from"react/jsx-runtime";function T(e){let{title:t,footer:r,children:o,width:n,height:i,innerHeight:s,isModal:a,backgroundColor:l}=e,{width:c}=g(),m=n??c,f=xo(null),b=ho(()=>{if(!f.current)return 0;let{width:d}=go(f.current);return d},[f.current]),p=b-8,S={...e,height:s,width:b},{activeSection:x}=w(),u=x===t||a?"cyan":"";return z(Qe,{ref:f,flexDirection:"column",height:i,width:m,backgroundColor:l,children:b!==0&&yo(vo,{children:[t&&z(ze,{color:u,wrap:"truncate-end",children:"\u256D\u2500\u2500\u2500\u2500 "+t+" "+"\u2500".repeat(p-t.length)+"\u256E"}),z(Qe,{flexDirection:"column",...S,borderStyle:"round",borderColor:u,borderTop:!t,borderBottom:!r,paddingLeft:1,children:o}),r&&z(ze,{color:u,wrap:"truncate-end",children:"\u2570"+"\u2500".repeat(p-r.length)+" "+r+" \u2500\u2500\u2500\u2500\u256F"})]})})}import{useEffect as bo,useState as ce}from"react";function y(e,t=[]){let[r,o]=ce(void 0),[n,i]=ce(!1),[s,a]=ce(0),l=()=>a(c=>c+1);return bo(()=>{let c=!1;return i(!1),o(void 0),e().then(m=>{c||(o(m),i(!0))}).catch(()=>{c||i(!0)}),()=>{c=!0}},[s,...t]),{value:r,resolved:n,refresh:l}}import{jsx as le}from"react/jsx-runtime";function Je(){let{width:e,sectionHeight:t}=g(),{value:r}=y(Ie);return le(So,{width:e,height:t,children:le(T,{title:"Status",innerHeight:t-1,children:r?.map(o=>le(wo,{children:o},o))})})}import{useEffect as Mt,useState as Pt}from"react";import{Box as vr,useInput as yr}from"ink";import{Text as To}from"ink";import{jsx as Xe}from"react/jsx-runtime";function Ze({sha:e}){let{resolved:t,value:r}=y(()=>Mo(e),[e]);return Xe(T,{title:"Commit Details",width:"50%",height:"100%",children:e&&t&&Xe(To,{children:r})})}async function Mo(e){if(!e)return;let{stdout:t}=await P(`git show ${e} --name-only`);return t.trim()}import{useCallback as Io,useEffect as Oo}from"react";import{Box as J,Text as ue,useInput as Ko}from"ink";import{useEffect as je,useRef as Po,useState as de}from"react";function O(e,t){let[r,o]=de(0),[n,i]=de(0),s=Po(void 0),[a,l]=de(0),c=()=>l(u=>u+1);je(()=>{let u=0;if(s.current!==void 0){let h=e.findIndex(v=>v===s.current);h>=0&&(u=h)}i(u);let d=Math.min(Math.max(u-Math.floor(t/2),0),Math.max(0,e.length-t));o(d)},[e,t,a]),je(()=>{s.current=e[n]},[n,e]);let m=Math.max(0,e.length-t),f=e.length-1,b=()=>{i(u=>{let d=Math.min(u+1,f);return o(h=>d>=h+t?Math.min(h+1,m):h),d})},p=()=>{i(u=>{let d=Math.max(u-1,0);return o(h=>d<h?Math.max(h-1,0):h),d})},S=e.slice(r,r+t),x=e[n];return{scrollDown:b,scrollUp:p,outList:S,selectedIndex:n,renderedIndex:n-r,selectedValue:x,refresh:c}}import{useInput as Co}from"ink";import{createContext as Eo,useContext as Ao}from"react";import{useEffect as rt,useState as et}from"react";import{jsx as Bo}from"react/jsx-runtime";var tt="\u2190",ot="\u2192",nt=Eo({});function it({children:e}){let[t,r]=et("truncate-end"),[o,n]=et(0),{removeKeybinding:i,setKeybinding:s}=M();return rt(()=>(o&&(t==="truncate-start"?s(tt,"pan left"):s(ot,"pan right")),()=>{i([ot,tt])}),[t,o]),Co((a,l)=>{o&&(l.rightArrow?r("truncate-start"):l.leftArrow&&r("truncate-end"))}),Bo(nt.Provider,{value:{mode:t,_isToggleable:o,_setIsToggleable:n},children:e})}function k(){let e=Ao(nt);if(Object.keys(e).length===0)throw new Error("useTruncationMode must be used within an TruncationModeProvider");let{mode:t,_setIsToggleable:r}=e;return rt(()=>(r(o=>o+1),()=>r(o=>o-1)),[]),{mode:t}}import{createContext as ko,useContext as Do,useEffect as st,useMemo as Lo,useRef as at,useState as ct}from"react";import{useInput as Ro}from"ink";import{jsx as Wo}from"react/jsx-runtime";var lt=ko({});function dt({children:e}){let{activeSection:t}=w(),{sectionHeight:r}=g(),o=r-3,{value:{tree:n,files:i}={},resolved:s}=y(No),a=at(new Set),l=at(new Set(["."])),[c,m]=ct();st(()=>{n&&i&&(a.current=new Set(Object.keys(n)),m(i[0]))},[s]);let f=Lo(()=>i?.filter(x=>a.current.has(x))??[],[a.current.size,s]),[b,p]=ct(0),S=()=>p(x=>x+1);return Ro((x,u)=>{if(t!=="Files"||!n||!c)return;if(u.upArrow){let h=Math.max(0,f.indexOf(c)-1);m(f[h]),S()}else if(u.downArrow){let h=Math.min(f.length-1,f.indexOf(c)+1);m(f[h]),S()}let d;if(u.leftArrow?d="collapse":u.rightArrow&&(d="expand"),d){let h=me(n,c);if(!h)return;for(let v of Object.keys(h)){let E=`${c}/${v}`;d==="expand"?a.current.add(E):a.current.delete(E)}S()}}),st(()=>{if(l.current.clear(),l.current.add("."),f.length<=o){f.forEach(_=>l.current.add(_));return}let x=c?Fo(c):[],u=Math.max(1,o-x.length),d=f.filter(_=>!x.includes(_)),h=c?d.indexOf(c):0,v=Math.max(0,h-Math.floor(u/2)),E=v+u;E>d.length&&(E=d.length,v=Math.max(0,E-u));let q=d.slice(v,E);[...x,...q].forEach(_=>{l.current.add(_)})},[f.length,c,o]),Wo(lt.Provider,{value:{version:b,tree:n,visibleFiles:l.current,expandedFiles:a.current,selectedFile:c},children:e})}function B(){let e=Do(lt);if(Object.keys(e).length===0)throw new Error("useTreeNavigation must be used within an TreeNavigationProvider");return e}var me=(e,t)=>{let r=t.split("/"),o=e;for(let n of r)o=o[n];return o},No=async()=>{let e=await Re(),t=Ne(e);return{tree:e,files:t}},Fo=e=>{let t=e.split("/"),r=[];for(let o=1;o<t.length;o++)r.push(t.slice(0,o).join("/"));return r};import{jsx as R,jsxs as Go}from"react/jsx-runtime";function mt({setSha:e,mode:t}){let{label:r,value:o}=t,{sectionHeight:n}=g(),{activeSection:i,isLocked:s}=w(),{selectedFile:a}=B(),{mode:l}=k(),c=Io(()=>o.length&&a?o(a):o(),[o,a]),{resolved:m,value:f=[]}=y(c,[r]),{scrollUp:b,scrollDown:p,outList:S,selectedValue:x,renderedIndex:u}=O(f,n-2),d=x?.sha;Oo(()=>{e(d)},[d]),Ko((v,E)=>{i!=="Log"||s||(E.upArrow?b():E.downArrow&&p())});let h=$o.includes(t.label);return R(T,{overflowY:"hidden",height:"100%",title:"Log",width:"50%",children:m&&S.map(({sha:v,message:E},q)=>Go(J,{flexDirection:"row",flexWrap:"nowrap",children:[R(J,{minWidth:2,children:v===d?R(ue,{color:h&&q===u?"magenta":void 0,children:"> "}):null}),R(J,{minWidth:8,children:R(ue,{color:"yellow",children:v})}),R(J,{children:R(ue,{wrap:l,children:E})})]},`${v}-${t.label}${h?q:""}`))})}var $o=["blame","reflog"];import{useState as rr}from"react";import{useInput as nr}from"ink";import ir from"ink-text-input";import{useInput as Ho}from"ink";import{createContext as Vo,useContext as _o,useEffect as Uo,useState as pe}from"react";import{Fragment as Yo,jsx as Qo,jsxs as qo}from"react/jsx-runtime";var ut=Vo({});function C(){let e=_o(ut);if(Object.keys(e).length===0)throw new Error("useKeybindings must be used within a KeybindingsProvider");return e}function pt({children:e}){let[t,r]=pe(null),{lock:o,unlock:n}=w(),[i,s]=pe(!1),[a,l]=pe(),c=()=>{n(),s(!1),a?.(),r(null)},m=()=>{o(),s(!0)},f=()=>{i?c():m()},b=(x,u)=>{c(),l(u?.onClose),r(x)};Ho((x,u)=>{u.escape&&c()});let{setKeybinding:p,removeKeybinding:S}=M();return Uo(()=>{if(i)return p("esc","close"),()=>{S("esc")}},[i]),Qo(ut.Provider,{value:{setModal:b,open:m,close:c,toggle:f,isOpen:i},children:qo(Yo,{children:[e,t]})})}import{Box as xt}from"ink";import fe from"node:path";import ge from"node:os";import zo from"node:fs/promises";import{createContext as Jo,useContext as Xo}from"react";import{jsx as tr}from"react/jsx-runtime";var ft=Jo({});function gt({children:e}){let{value:t}=y(Zo);return tr(ft.Provider,{value:{bgColor:t},children:e})}var ht=()=>{let e=Xo(ft);if(Object.keys(e).length===0)throw new Error("useKeybindings must be used within a KeybindingsProvider");return e};async function Zo(){if(!(!process.stdin.isTTY||!process.stdout.isTTY))return process.env.TERM_PROGRAM==="vscode"?await er():new Promise(e=>{let t="\x1B]11;?\x07",r="",o,n=s=>{clearTimeout(o),process.stdin.off("data",i);try{process.stdin.setRawMode(!1),process.stdin.pause()}catch{}e(s)},i=s=>{r+=s.toString("ascii");let a=r.match(/\x1b]11;rgb:([0-9a-fA-F]{1,4})\/([0-9a-fA-F]{1,4})\/([0-9a-fA-F]{1,4})\x07/);if(a){let l=jo(a[1],a[2],a[3]);n(l)}};try{process.stdin.setRawMode(!0),process.stdin.resume(),process.stdin.on("data",i),o=setTimeout(()=>n(void 0),150),process.stdout.write(t)}catch{n(void 0)}})}function jo(e,t,r){return"#"+[e,t,r].map(o=>Math.round(parseInt(o,16)*255/((1<<4*o.length)-1)).toString(16).padStart(2,"0")).join("")}async function er(){let e="Code";process.env.TERM_PROGRAM_VERSION?.includes("insider")&&(e="Code - Insiders");let t;switch(process.platform){case"win32":t=fe.join(ge.homedir(),"AppData","Roaming",e,"User","settings.json");break;case"linux":t=fe.join(ge.homedir(),".config",e,"User","settings.json");break;case"darwin":t=fe.join(ge.homedir(),"Library","Application Support",e,"User","settings.json");break;default:return}if(!await Le(t))return;let r=await zo.readFile(t,{encoding:"utf-8"}),{default:o}=await import("strip-json-comments"),n=o(r).replace(/,\s*(\}|\])/g,"$1");switch(JSON.parse(n)["workbench.colorTheme"]){case"Default Dark Modern":case void 0:return"#181818";case"Visual Studio Dark":case"Default Dark+":return"#1e1e1e";case"Default High Contrast":return"#000000";case"Default Light+":case"Visual Studio Light":case"Default High Contrast Light":return"#ffffff";case"Default Light Modern":return"#f8f8f8";default:return}}import{jsx as he}from"react/jsx-runtime";var or=36;function K({children:e,title:t,width:r=or}){let o=g(),{width:n}=o,i=Math.floor((n-r)/2),{bgColor:s}=ht();return he(xt,{...o,position:"absolute",children:he(xt,{alignSelf:"center",marginLeft:i,marginRight:i,children:he(T,{title:t,width:r,isModal:!0,backgroundColor:s,children:e})})})}import{jsx as vt}from"react/jsx-runtime";function X({title:e,handleSubmit:t}){let[r,o]=rr(""),{isOpen:n,close:i}=C();return nr((s,a)=>{a.return&&n&&(t(r),o(""),i())}),n&&vt(K,{title:e,children:vt(ir,{value:r,onChange:o})})}import{useInput as sr}from"ink";import ar from"ink-select-input";import{jsx as yt}from"react/jsx-runtime";function Z({title:e,handleSubmit:t,options:r,initialIndex:o,width:n}){let{isOpen:i,close:s}=C();return sr((a,l)=>{l.return&&i&&s()}),i&&yt(K,{title:e,width:n,children:yt(ar,{items:r,onSelect:t,initialIndex:o})})}var G=[{value:()=>$("git log --oneline -n 400"),label:"log"},{value:()=>$("git log --oneline --tags --no-walk --decorate"),label:"tags"},{value:()=>$("git reflog"),label:"reflog"},{value:()=>$('git log --oneline --author="$(git config user.name)"'),label:"my commits"},{value:e=>$(`git log --oneline "${e}"`),label:"file"},{value:e=>$(`git blame --follow "${e}"`),label:"blame"}];async function $(e){let{stdout:t}=await P(e);return t.trim().split(/\r?\n/).map(cr)}function cr(e){let t=e.indexOf(" ");return{sha:e.slice(0,t),message:e.slice(t+1)}}import{createContext as lr,useContext as dr,useEffect as mr,useState as ur}from"react";import{jsx as pr}from"react/jsx-runtime";var bt=lr({});function St({children:e}){let[t,r]=ur(),o=(n,...i)=>{n(...i).catch(s=>r(s.stderr??s.message))};return mr(()=>{if(t)throw new Error(t)},[t]),pr(bt.Provider,{value:{attempt:o,err:t},children:e})}function H(){let e=dr(bt);if(Object.keys(e).length===0)throw new Error("useErrorCatcher must be used within an ErrorCatcherProvider");return e}import{createContext as fr,useContext as gr,useState as hr}from"react";import{jsx as xr}from"react/jsx-runtime";var wt=fr({});function Tt({children:e}){let[t,r]=hr(),o=i=>r(i),n=i=>{let s=t===i;return s&&r(void 0),s};return xr(wt.Provider,{value:{sendMessage:o,receiveMessage:n},children:e})}function j(){let e=gr(wt);if(Object.keys(e).length===0)throw new Error("useMessaging must be used within an MessagingProvider");return e}import{jsx as xe,jsxs as br}from"react/jsx-runtime";function ve(){let{setModal:e,toggle:t}=C(),[r,o]=Pt(),{width:n,sectionHeight:i}=g(),[s,a]=Pt(G[0]),{receiveMessage:l}=j();Mt(()=>{l("log-file")&&a(G[4]),l("blame-file")&&a(G[5])},[]);let{isLocked:c}=w(),{attempt:m}=H();yr((p,S)=>{p==="m"&&(e(xe(Z,{options:G,title:"Log Type",handleSubmit:x=>a(x),initialIndex:G.indexOf(s)})),t()),!c&&p==="c"&&r&&m(()=>P(`git checkout ${r}`))});let{setKeybinding:f,removeKeybinding:b}=M();return Mt(()=>(f("c","checkout"),f("m","mode"),()=>{b(["c","m"])}),[]),br(vr,{width:n,height:i,children:[xe(mt,{setSha:o,mode:s}),xe(Ze,{sha:r})]})}import{useEffect as Br}from"react";import{Box as Dt}from"ink";import{Box as ye,Text as Ct,useInput as Sr}from"ink";import{useEffect as wr}from"react";import{Fragment as Mr,jsx as N,jsxs as Tr}from"react/jsx-runtime";function Et(){let{activeSection:e,isLocked:t}=w(),{sectionHalfHeight:r}=g(),{value:o=[],refresh:n}=y(Ke),{value:i,refresh:s}=y(Fe),{setKeybinding:a,removeKeybinding:l}=M();wr(()=>{if(e==="Branches")return a("s","switch"),()=>{l("s")}},[e]);let{outList:c,selectedValue:m,scrollUp:f,scrollDown:b}=O(o,r-2),p=async v=>{await P(`git branch ${v}`),n()},S=async()=>{await P(`git branch -D ${m}`),n()},x=async()=>{await P(`git switch ${m}`),n(),s()},{setModal:u,open:d}=C(),{attempt:h}=H();return Sr((v,E)=>{e!=="Branches"||t||(E.upArrow?f():E.downArrow?b():v==="c"?(u(N(X,{title:"Name your new branch",handleSubmit:p})),d()):v==="d"?h(S):v==="s"&&h(x))}),N(Mr,{children:N(T,{width:"100%",title:"Branches",innerHeight:r-1,children:c.map(v=>Tr(ye,{flexDirection:"row",flexWrap:"nowrap",children:[N(ye,{minWidth:2,children:v===m?N(Ct,{children:"> "}):null}),N(ye,{children:N(Ct,{color:v===i?"magenta":void 0,children:v})})]},v))})})}import{Text as be}from"ink";import{Fragment as Pr}from"react";import{jsx as Se,jsxs as At}from"react/jsx-runtime";function Bt(){let{value:e}=y($e);return Se(T,{width:"50%",title:"Remotes",children:e?.map(({url:t,name:r,type:o})=>At(Pr,{children:[At(be,{wrap:"truncate-end",children:[r," (",o,")"]}),Se(be,{wrap:"truncate-end",children:t}),Se(be,{children:" "})]},`${r}-${o}`))})}import{Text as we}from"ink";import{Fragment as Cr}from"react";import{jsx as ee,jsxs as Ar}from"react/jsx-runtime";function kt(){let{sectionHalfHeight:e}=g(),{value:t}=y(Er);return ee(T,{width:"100%",title:"Worktrees",innerHeight:e-1,children:t?.map(({branch:r,detached:o,path:n,head:i})=>Ar(Cr,{children:[ee(we,{children:o?i:r}),ee(we,{children:n}),ee(we,{children:" "})]},n))})}async function Er(){let{stdout:e}=await P("git worktree list --porcelain"),t=e.trim().split(/\r?\n/),r=[],o=[];for(let n of t)n.trim()===""?o.length&&(r.push(o),o=[]):o.push(n);return o.length&&r.push(o),r.map(n=>{let i={detached:!1};for(let s of n)s.startsWith("worktree ")?i.path=s.slice(9):s.startsWith("HEAD ")?i.head=s.slice(5):s.startsWith("branch ")?i.branch=s.slice(7):s==="detached"&&(i.detached=!0);return i})}import{Fragment as kr,jsx as te,jsxs as Lt}from"react/jsx-runtime";function Rt(){let{width:e,sectionHeight:t}=g(),{setKeybinding:r,removeKeybinding:o}=M();return Br(()=>(r("c","create"),r("d","delete"),()=>{o(["c","d"])}),[]),te(kr,{children:Lt(Dt,{width:e,height:t,children:[Lt(Dt,{flexDirection:"column",width:"50%",height:t,children:[te(Et,{}),te(kt,{})]}),te(Bt,{})]})})}import{Component as Dr,useEffect as Lr}from"react";import{Text as Nt}from"ink";import{jsx as Te}from"react/jsx-runtime";function oe({title:e,message:t,width:r}){let{isOpen:o}=C(),{mode:n}=k();return o&&Te(K,{title:e,width:r,children:Array.isArray(t)?t.map(i=>Te(Nt,{wrap:n,children:i})):Te(Nt,{children:t})})}import{jsx as Ft}from"react/jsx-runtime";var re=class extends Dr{state={hasError:!1};constructor(t){super(t)}static getDerivedStateFromError(t){return{hasError:!0,message:t?.stderr??t?.message}}dismissError(){this.setState({hasError:!1})}componentDidCatch(t,r){let o=t?.stderr??t?.message;this.setState({hasError:!0,message:o})}render(){return this.state.hasError?Ft(Rr,{dismissError:this.dismissError.bind(this),message:this.state.message,children:this.props.children}):this.props.children}};function Rr({message:e,dismissError:t,children:r}){let{setModal:o,open:n}=C(),{modalWidth:i}=g();return Lr(()=>{if(e){let s=e.split(/\r?\n/).map(a=>a.trim());o(Ft(oe,{title:"Error",message:s,width:i}),{onClose:t}),n()}},[e]),r}import{Box as Ur,useInput as Yr}from"ink";import{useEffect as qr}from"react";import{Box as Me,Text as Kt}from"ink";import Pe from"node:path";import{memo as Nr,useEffect as Wt,useState as Fr}from"react";import{Fragment as Kr,jsx as D,jsxs as Ce}from"react/jsx-runtime";var Wr="\u2190",Ir="\u2191",$t="\u2192",Gt="\u2193",It=`${Wr}${Ir}${Gt}${$t}`;function Ht(){let{sectionHeight:e}=g(),{activeSection:t}=w(),{setKeybinding:r,removeKeybinding:o}=M();Wt(()=>{if(t==="Files")return r(It,"NAVIGATION_KEY"),()=>o(It)},[t]);let{tree:n,version:i}=B(),{value:s}=y(We),[,a]=Fr(0);return Wt(()=>{a(l=>l+1)},[i]),D(T,{title:"Files",innerHeight:e,width:"50%",children:s&&n&&D(Vt,{name:Pe.basename(s),contents:n,depth:0,fp:s})})}function Vt(e){let{contents:t,depth:r,fp:o,rp:n="."}=e,i=Object.keys(t).length===0,{selectedFile:s,visibleFiles:a,expandedFiles:l}=B(),c=n===s;if(!a.has(n))return null;let m=r+2;if(i)return D(Ot,{...e,depth:m,isSelected:c},o);let f=`${n}/`,b=Or(l,p=>p.startsWith(f))||r===0;return Ce(Kr,{children:[Ce(Me,{flexDirection:"row",flexWrap:"nowrap",children:[D(Me,{width:2,children:D(Kt,{children:b?Gt:$t})}),D(Me,{width:"100%",children:D(Ot,{...e,isSelected:c})})]}),b&&Object.entries(t).map(([p,S])=>{let x=Pe.join(n,p),u=Pe.join(o,p);return D(Vt,{name:p,contents:S,depth:m,rp:x,fp:u})})]})}var Ot=Nr(({name:e,depth:t,isSelected:r})=>Ce(Kt,{wrap:"truncate-end",color:r?"yellow":void 0,children:["\xA0".repeat(t),e]}),(e,t)=>e.isSelected===t.isSelected&&e.fp===t.fp);function Or(e,t){for(let r of e)if(t(r))return!0;return!1}import{Box as Ut,Text as F}from"ink";import{Box as $r,measureElement as Gr,Text as _t}from"ink";import{useMemo as Hr,useRef as Vr}from"react";import{jsx as Ee}from"react/jsx-runtime";function ne({text:e}){let t=Vr(null),r=Hr(()=>{if(!t.current)return 0;let{width:a}=Gr(t.current);return a-4},[t.current]),o=Math.max(0,r-e.length),n=Math.floor(o/2),i=o-n,s=`${"=".repeat(n)} ${e} ${"=".repeat(i)}`;return Ee($r,{width:"100%",ref:t,children:e.length>r?Ee(_t,{children:e}):Ee(_t,{children:s})})}import{jsx as Ae,jsxs as L}from"react/jsx-runtime";function Yt(){let{sectionHeight:e}=g(),{selectedFile:t}=B(),{value:r={}}=y(()=>Ge(t),[t]),{authorDate:o,authorEmail:n,authorName:i,shortHash:s}=r,{value:{added:a,deleted:l}={},resolved:c}=y(async()=>_r(s,t),[s,t]),{mode:m}=k();return Ae(Ut,{width:"50%",height:e,children:L(T,{width:"100%",title:"File Meta",innerHeight:e-1,children:[t&&Ae(ne,{text:t}),s&&L(F,{wrap:m,children:["Last Commit: ",s]}),i&&L(F,{wrap:m,children:["Edited By: ",i]}),o&&L(F,{wrap:m,children:["Edit On: ",o]}),n&&L(F,{wrap:m,children:["Contact: ",n]}),c&&(a||l)&&L(Ut,{flexDirection:"row",flexWrap:"nowrap",children:[Ae(F,{children:"Changes:\xA0"}),L(F,{color:"green",children:["+",a??0,"\xA0"]}),L(F,{color:"red",children:["-",l??0]})]})]})})}var _r=async(e,t)=>!e||!t?{}:(await Q(e)).find(({file:o})=>t===o)??{};import{jsx as Be,jsxs as Qr}from"react/jsx-runtime";function qt(){let{width:e,sectionHeight:t}=g(),{selectedFile:r,tree:o}=B(),{setKeybinding:n,removeKeybinding:i}=M();qr(()=>(n("l","log"),n("b","blame"),()=>{i(["l","b"])}),[]);let{setActiveGroup:s}=w(),{sendMessage:a}=j(),{setModal:l,open:c}=C();return Yr((m,f)=>{m==="l"?(a("log-file"),s("Log")):m==="b"&&(Object.keys(me(o??{},r)).length===0?(a("blame-file"),s("Log")):(l(Be(oe,{title:"Sorry",message:["Can't blame a folder."]})),c()))}),Qr(Ur,{width:e,height:t,children:[Be(Ht,{}),Be(Yt,{})]})}import{Box as on}from"ink";import{Box as Qt,Text as zt,useInput as zr}from"ink";import{useEffect as Jr}from"react";import{jsx as V,jsxs as Zr}from"react/jsx-runtime";function Jt({setStash:e}){let{sectionHeight:t}=g(),{mode:r}=k(),{attempt:o}=H(),{setModal:n,open:i}=C(),{activeSection:s,isLocked:a}=w(),{value:l=[],refresh:c}=y(Xr),{outList:m,selectedValue:f,selectedIndex:b,scrollUp:p,scrollDown:S}=O(l,t-2);Jr(()=>{e(`stash@{${b}}`)},[b]);let x=d=>async()=>{await P(`git stash ${d} stash@{${b}}`),c()},u=async()=>{let d=`stash@{${b}}`,h=await Oe(d);n(V(X,{title:`reword ${d}`,handleSubmit:async v=>{await P(`git stash drop ${d}`),await P(`git stash store ${h} -m "${v}"`),c()}})),i()};return zr((d,h)=>{a||(d==="s"?(n(V(Z,{title:"Stash",options:[{value:"git stash",label:"tracked only"},{value:"git stash -u",label:"tracked & untracked"}],handleSubmit:({value:v})=>o(async()=>{await P(v),c()})})),i()):d==="a"?o(x("apply")):d==="p"?o(x("pop")):d==="d"?o(x("drop")):d==="r"&&o(u),s==="Stashes"&&(h.upArrow?p():h.downArrow&&S()))}),V(T,{title:"Stashes",width:"50%",height:"100%",children:m.map(d=>Zr(Qt,{flexDirection:"row",flexWrap:"nowrap",children:[V(Qt,{minWidth:2,children:d===f?V(zt,{children:"> "}):null}),V(zt,{wrap:r,children:d})]},d))})}async function Xr(){return await De("git stash list")}import{Box as jr,Text as ke}from"ink";import{jsx as tn,jsxs as Y}from"react/jsx-runtime";function Xt({stash:e}){let{value:t=[]}=y(()=>en(e),[e]);return Y(T,{title:"Stash Details",width:"50%",height:"100%",children:[e&&tn(ne,{text:e}),t.map(({file:r,added:o,deleted:n,changeType:i})=>i!=="binary"&&Y(jr,{flexDirection:"row",flexWrap:"nowrap",children:[Y(ke,{children:[r,":\xA0"]}),Y(ke,{color:"green",children:["+",o??0,"\xA0"]}),Y(ke,{color:"red",children:["-",n??0]})]}))]})}async function en(e){if(e)return await Q(e)}import{useEffect as rn,useState as nn}from"react";import{jsx as Zt,jsxs as sn}from"react/jsx-runtime";function jt(){let{sectionHeight:e,width:t}=g(),[r,o]=nn(),{setKeybinding:n,removeKeybinding:i}=M();return rn(()=>(n("s","stash"),n("d","drop"),n("a","apply"),n("p","pop"),n("r","reword"),()=>i(["s","p","d","a","r"])),[]),sn(on,{width:t,height:e,children:[Zt(Jt,{setStash:o}),Zt(Xt,{stash:r})]})}import{jsx as A,jsxs as dn}from"react/jsx-runtime";function eo(){let{prevSection:e,nextSection:t,activeGroup:r,setActiveGroup:o}=w();return cn((n,i)=>{i.tab&&i.shift?e():i.tab?t():n==="1"?o("Files"):n==="2"?o("Status"):n==="3"?o("Log"):n==="4"?o("Branches"):n==="5"&&o("Stashes")}),A(re,{children:A(St,{children:A(Tt,{children:A(dt,{children:A(ln,{group:r})})})})})}function ln({group:e}){switch(e){case"Files":return A(qt,{});case"Status":return A(Je,{});case"Log":return A(ve,{});case"Branches":return A(Rt,{});case"Stashes":return A(jt,{});default:return dn(an,{children:['Error, unknown page "',e,'"']})}}import{jsx as ie,jsxs as fn}from"react/jsx-runtime";function oo(){let{isLocked:e,activeGroup:t}=w();pn((i,s)=>{e||i.toLowerCase()==="q"&&process.exit()});let r=g(),{keybindings:o,setKeybinding:n}=M();return mn(()=>n("q","quit"),[]),fn(to,{flexDirection:"column",...r,children:[ie(qe,{activeGroup:t}),ie(eo,{}),ie(to,{flexDirection:"row",children:Object.entries(o).map(([i,s])=>ie(un,{children:`[${i}]: ${s} `},s))})]})}import{jsx as W}from"react/jsx-runtime";function hn(){return W(gt,{children:W(Ue,{children:W(Ve,{children:W(it,{children:W(pt,{children:W(oo,{})})})})})})}var vl=()=>gn(W(hn,{}),{patchConsole:!0});export{vl as renderApp};
|
package/dist/main.js
CHANGED
|
@@ -1,2 +1,2 @@
|
|
|
1
1
|
#!/usr/bin/env node
|
|
2
|
-
import{a as o,d as t}from"./chunks/chunk-
|
|
2
|
+
import{a as o,d as t}from"./chunks/chunk-HKY7QTE6.js";import"./chunks/chunk-7XT4D6II.js";import"./chunks/chunk-R7DNYVL6.js";async function a(){let r=o();if(!r){t();let{renderApp:i}=await import("./chunks/pages-YLGVYIE5.js");return i()}let{runCommand:n}=await import("./chunks/commands-6NWVWJQB.js");if(await n(r))return;let{runWrapper:e}=await import("./chunks/wrapper-PHXQ7CVF.js");await e(r)||(console.error("unknown command"),process.exit(1))}a().catch(r=>{console.error(`An error occurred: ${r.stderr??r.message}`),process.exit(1)});
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "bisgit",
|
|
3
|
-
"version": "0.
|
|
3
|
+
"version": "0.8.0",
|
|
4
4
|
"description": "Git CLI tool to simplify common advanced git workflows",
|
|
5
5
|
"main": "dist/main.js",
|
|
6
6
|
"bin": {
|
|
@@ -41,10 +41,10 @@
|
|
|
41
41
|
"prepare": "husky"
|
|
42
42
|
},
|
|
43
43
|
"devDependencies": {
|
|
44
|
-
"@types/node": "^25.0.
|
|
45
|
-
"@types/react": "^19.2.
|
|
46
|
-
"@typescript-eslint/eslint-plugin": "^8.53.
|
|
47
|
-
"@typescript-eslint/parser": "^8.53.
|
|
44
|
+
"@types/node": "^25.0.10",
|
|
45
|
+
"@types/react": "^19.2.9",
|
|
46
|
+
"@typescript-eslint/eslint-plugin": "^8.53.1",
|
|
47
|
+
"@typescript-eslint/parser": "^8.53.1",
|
|
48
48
|
"esbuild": "^0.27.2",
|
|
49
49
|
"esbuild-lazy-analyzer": "^1.4.0",
|
|
50
50
|
"esbuild-visualizer": "^0.7.0",
|
|
@@ -53,12 +53,12 @@
|
|
|
53
53
|
"husky": "^9.1.7",
|
|
54
54
|
"lint-staged": "^16.2.7",
|
|
55
55
|
"npm-run-all": "^4.1.5",
|
|
56
|
-
"prettier": "^3.8.
|
|
56
|
+
"prettier": "^3.8.1",
|
|
57
57
|
"react-devtools-core": "^6.1.5"
|
|
58
58
|
},
|
|
59
59
|
"dependencies": {
|
|
60
60
|
"chalk": "^5.6.2",
|
|
61
|
-
"clipboardy": "^5.0
|
|
61
|
+
"clipboardy": "^5.1.0",
|
|
62
62
|
"ink": "^6.6.0",
|
|
63
63
|
"ink-select-input": "^6.2.0",
|
|
64
64
|
"ink-spinner": "^5.0.0",
|
|
@@ -1,2 +0,0 @@
|
|
|
1
|
-
#!/usr/bin/env node
|
|
2
|
-
import{b as n,c as a,d as u}from"./chunk-R7DNYVL6.js";import l from"fs/promises";async function $(t){try{return await l.access(t),!0}catch{return!1}}async function f(){return await a("git ls-files")}async function p(){return await a("git ls-files --others --exclude-standard")}async function P(){let[t,r]=await Promise.all([f(),p()]),e=[...t,...r].sort(([o],[s])=>o.localeCompare(s)),i={};for(let o of e){let s=o.split("/"),c=i;for(let m of s)c[m]||(c[m]={}),c=c[m]}return i}function h(t,r=""){let e=[];for(let[i,o]of Object.entries(t)){let s=r?`${r}/${i}`:i;e.push(s),Object.keys(o).length>0&&e.push(...h(o,s))}return e}import{normalize as g}from"node:path";async function d(){let{stdout:t}=await n("git branch --show-current");return t.trim()}async function B(t){t??=await d();let{stdout:r}=await n(`git config --get branch.${t}.remote`);return r.trim()}async function D(t,r){try{let{stdout:e}=await n(`git rev-list --count ${t}/${r}..${r}`);return parseInt(e.trim(),10)}catch(e){return console.error("Error getting commits ahead:",e),NaN}}async function T(t,r){try{let{stdout:e}=await n(`git rev-list --count ${r}..${t}/${r}`);return parseInt(e.trim(),10)}catch(e){return console.error("Error getting commits behind:",e),NaN}}async function E(t){let{stdout:r}=await n(`git rev-parse --git-path ${t}`);return g(r.trim())}async function R(){let{stdout:t}=await n("git rev-parse --git-dir");return g(t.trim())}async function N(){let{stdout:t}=await n("git rev-parse --show-toplevel");return g(t.trim())}async function S(t){let{code:r}=await u("git",["remote","get-url",t],{stdio:"ignore"});return r===0}async function G(){let{stdout:t}=await n("git rev-parse --abbrev-ref HEAD");return t.trim()==="HEAD"}async function I(t,r){let e="git fetch";t&&(e+=` ${t}`),r&&(e+=` ${r}`),await n(e)}async function M(){let{stdout:t}=await u("git",["status","--porcelain"]);return!t?.trim()}async function k(){return await a("git status --porcelain")}async function y(t,r){let{stdout:e}=await n(`git merge-base ${t} ${r}`);return e.trim()}async function z(t,r){let e=await y(t,r),{stdout:i}=await u("git",["merge-tree",e,r,t]);return!i?.split(/\r?\n/)?.includes("+=======")}async function A(t){let{stdout:r}=await n(`git rev-parse ${t}`);return r.trim()}async function F(t,r,e){return await a(`git rev-list ${e?"--reverse":""} ${t}..${r}`)}async function H(){return await a("git for-each-ref --format='%(refname:short)' refs/heads")}async function C(){let{stdout:t}=await n("git remote -v");return t.trim().split(/\r?\n/).map(e=>{let i=e.match(/^(\S+)\s+(\S+)\s+\((fetch|push)\)$/);if(!i)return null;let[,o,s,c]=i;return{name:o,url:s,type:c}}).filter(e=>e!==null)}async function O(t){try{let{stdout:r}=await n(`git log -1 --format='%an|%ae|%ad|%h' -- "${t}"`,{encoding:"utf8"}),[e,i,o,s]=r.trim().split("|");return{authorName:e,authorEmail:i,authorDate:o,shortHash:s}}catch{return{}}}export{$ as a,P as b,h as c,d,B as e,D as f,T as g,E as h,R as i,N as j,S as k,G as l,I as m,M as n,k as o,y as p,z as q,A as r,F as s,H as t,C as u,O as v};
|
|
@@ -1,12 +0,0 @@
|
|
|
1
|
-
#!/usr/bin/env node
|
|
2
|
-
import{b as g,c as K,d as rt,e as R,f as d,g as I,h as C,i as H,j as M,k as it,l as y,m as A}from"./chunk-L4626BRA.js";import{a as $,d as h,e as z,f as B,g as q,h as D,i as Z,j as Q,k as X,m as W,n as x,o as G,p as P,q as tt,r as et,s as F,t as ot}from"./chunk-DLHREY2Q.js";import{a as u,b as s,d as w,e as l}from"./chunk-R7DNYVL6.js";import S from"node:path";async function nt(){let t=await G(),e=/^(UU|DU|UD)/;return!!t?.some(o=>e.test(o))}async function N(){let t=await Z(),[e,o,i,r,n]=await Promise.all([await $(S.join(t,"MERGE_HEAD")),await $(S.join(t,"rebase-merge")),await $(S.join(t,"rebase-apply")),await $(S.join(t,"CHERRY_PICK_HEAD")),await $(S.join(t,"REVERT_HEAD"))]);return{merge:e,rebaseMerge:o,rebaseApply:i,cherryPick:r,revert:n}}async function st(){let{merge:t,rebaseMerge:e,rebaseApply:o,cherryPick:i,revert:r}=await N(),n="";if(t)n="merge";else if(e||o)n="rebase";else if(i)n="cherry-pick";else if(r)n="revert";else{let a=await nt()?"Conflicts exist, but likely are from a stash and are not abortable. Resolve them manually.":"Nothing to abort.";console.log(a);return}console.log(`Aborting the ${n}`),await l([n,"--abort"])}async function at(){await M(),await A(),u("git",["commit","--amend","--no-edit",...g()],{stdio:"inherit",silent:!0,triggerExit:!0})}import{render as Yt}from"ink";import{render as _t}from"ink";import{Suspense as Nt,use as Tt,useEffect as Lt}from"react";import{Text as ct,useApp as Ut}from"ink";import It from"ink-spinner";import{jsx as T,jsxs as jt}from"react/jsx-runtime";function c(t){let{msg:e}=t;return T(Nt,{fallback:jt(ct,{children:[T(It,{type:"dots"}),"\xA0",e]}),children:T(Ot,{...t})})}function Ot({msg:t,promise:e}){let o=Tt(e),{exit:i}=Ut();return Lt(()=>{i()},[o]),T(ct,{children:`\u2714 ${typeof o=="string"?o:t}`})}import{jsx as Vt}from"react/jsx-runtime";async function mt(){let t=process.argv[3],e=process.argv[4]??await h();if(I(t),I(e),t===e)return console.log(`No reason to merge '${t}' into '${e}'`);let o=await z(t);d(o),_t(Vt(c,{msg:"Updating target branch and merging",promise:(async()=>{try{await E(o,t)}catch{console.error(`Error when updating branch ${t} with remote ${o}. Fix divergent branches before merging into ${e}`);return}await h()!==e&&await s(`git switch ${e}`),u("git",["merge",t],{stdio:"inherit",silent:!0})})()}))}async function E(t,e){await W(t,e);let[o,i,r]=await Promise.all([B(t,e),q(t,e),h()]);if(i!==0){if(o===0&&r!==e){await s(`git fetch ${t} ${e}:${e}`);return}if(await s(`git switch ${e}`),o===0){await s(`git merge ${t}/${e}`);return}await l(["merge",`${t}/${e}`],["ignore","ignore","inherit"])}}async function k(){R(),d();let{stdout:t,code:e}=await w("git",["rev-parse","--abbrev-ref","origin/HEAD"]);if(e===0)return lt(t.trim());await s("git remote set-head origin -a");let{stdout:o}=await s("git rev-parse --abbrev-ref origin/HEAD");return lt(o.trim())}async function ft(){let{remote:t,branch:e}=await k();console.log(`Remote default branch is '${t}/${e}'`)}async function lt(t){let e=t.split("/"),o=t.split("/").map((i,r)=>e.slice(0,r+1).join("/"));for(let i of o)if(await X(i)){let r=t.slice(i.length+1);return{remote:i,branch:r}}throw new Error("Failed to get default remote")}import{jsx as Kt}from"react/jsx-runtime";var Jt=["master","main","development","lingoport"];async function pt(){await C();let{branch:t,remote:e}=await k();await s(`git switch ${t}`);let o=(async()=>{await l(["fetch","--prune","--prune-tags"]),await E(e,t)})();Yt(Kt(c,{msg:"Fetching and pruning remote references",promise:o})),await o;let i=await ot();for(let r of i){if(r===t||Jt.includes(r))continue;let{code:n}=await w("git",["diff","--quiet",t,r]);if(n===0){O(r);continue}let{code:a}=await w("git",["merge-base","--is-ancestor",r,t]);if(a===0){O(r);continue}let m=await P(r,t),{stdout:f}=await w("git",["merge-tree",m,t,r]);f?.trim()||O(r)}}function O(t){l(["branch","-D",t])}import{spawn as zt}from"node:child_process";import{createInterface as Zt}from"node:readline/promises";async function gt(){R;let e=[...(await Qt()).entries()].sort(([,i],[,r])=>r-i).slice(0,25),o=e[0][1].toString().length;console.info(e.map(([i,r])=>`${String(r).padEnd(o)} ${i}:`).join(`
|
|
3
|
-
`))}async function Qt(){let{promise:t,resolve:e,reject:o}=Promise.withResolvers(),i=new Map,r=zt("git",["log","--all","-M","-C","--name-only","--format=format:"]),n=Zt({input:r.stdout,crlfDelay:1/0});return n.on("line",a=>{let m=a.trim();m&&i.set(m,(i.get(m)??0)+1)}),n.on("close",()=>e(i)),r.on("error",o),t}import{render as Xt}from"ink";import{jsx as ee}from"react/jsx-runtime";async function ut(){await A();let t=y("Error: missing github username argument"),e=(async()=>{let r=await te(t);return r.email||(console.error(`Couldn't find an email for ${t}`),process.exit(1)),r})();Xt(ee(c,{msg:"Fetching name and email",promise:e}));let{name:o,email:i}=await e;l(["commit","--edit","-m","<insert summary>","-m",`Co-authored-by: ${o} <${i}>`])}async function te(t){let{stdout:e}=await s(`gh api users/${t}`);return JSON.parse(e.trim())}import{render as j}from"ink";import{isDeepStrictEqual as oe}from"node:util";import{jsx as _}from"react/jsx-runtime";async function wt(){let t=y("Error: missing id of PR to checkout");await C();let e=(async()=>Promise.all([s(`gh co ${t}`),s(`gh pr view ${t} --json baseRefName -q .baseRefName`)]))();j(_(c,{msg:`Checking out pr #${t}`,promise:e}));let[,{stdout:o}]=await e,i=o.trim(),r=(async()=>{await s(`git switch ${i}`),await s("git pull"),await s("git switch -")})();j(_(c,{msg:`Updating ${i} branch merge destination`,promise:r})),await r;let n=(async()=>{let p=await P(i,"HEAD");return await s(`git reset ${p} --soft`),await re("HEAD")})();j(_(c,{msg:"Creating an editable, explorable, local PR experience",promise:n}));let[a,m]=await Promise.all([n,ie(t)]),{files:f,additions:b,deletions:U}=m;if(oe(a,m))return console.info(`Files: ${f}, +${b} -${U}`);console.info("The PR stat doesn't match the local stat."),console.info("LOCAL"),console.table(a),console.info("GITHUB"),console.table(m),console.info("One option to fix (there'll likely be conflicts):"),console.info("1. git stash -u"),console.info(`2. git merge ${i}`),console.info("3. git stash pop")}async function re(t){let{stdout:e}=await s(`git diff --shortstat ${t}`),o=/(\d+)\s+files? changed(?:, (\d+) insertions?\(\+\))?(?:, (\d+) deletions?\(-\))?/,i=e.trim().match(o);if(!i)throw new Error(`Couldn't generate short stat for ref(s): ${t}`);let[,r,n,a]=i;return{files:parseInt(r,10),additions:n?parseInt(n,10):0,deletions:a?parseInt(a,10):0}}async function ie(t){let{stdout:e}=await s(`gh pr view ${t} --json deletions,files,additions`),o=JSON.parse(e.trim());return o.files=o.files.length,o}import{spawn as ne}from"node:child_process";import{createInterface as se}from"node:readline/promises";import V from"chalk";async function ht(){let t=y("Error: missing target branch argument"),e=await P("HEAD",t),{promise:o,resolve:i,reject:r}=Promise.withResolvers(),n=ne("git",["merge-tree",e,"HEAD",t]),a=se({input:n.stdout,crlfDelay:1/0}),m=!1,f=0;a.on("line",p=>{p=p.trimEnd(),p==="+<<<<<<< .our"?(m=!0,console.info(V.red(p)),f+=1):p==="+>>>>>>> .their"?(m=!1,console.info(V.green(p))):p==="+======="?console.info(V.yellow(p)):m&&console.info(p)}),a.on("close",()=>i(f)),n.on("error",r);let b=await o,U=b?`\u274C ${b} conflicts would occur!`:"\u2705 No conflicts. Safe to merge.";console.info(U)}async function dt(){let{merge:t,rebaseMerge:e,rebaseApply:o,cherryPick:i,revert:r}=await N(),n="";if(t)n="merge";else if(e||o)n="rebase";else if(i)n="cherry-pick";else if(r)n="revert";else{console.log("Nothing to continue.");return}console.log(`Continuing the ${n}`),await l([n,"--continue"])}import{appendFileSync as ce}from"node:fs";import{readFile as ae}from"node:fs/promises";async function L(t){return(await ae(t,"utf-8")).split(/\r?\n/).map(o=>o.trim()).filter(o=>o&&!o.startsWith("#"))}async function yt(){let t=await D("info/exclude");await H(t);let e=await L(t);g().forEach(o=>{e.includes(o)?console.info(`Already Exists: '${o}'`):(ce(t,`${o}
|
|
4
|
-
`),console.info(`Added '${o}'`))}),console.info(`
|
|
5
|
-
See file at ${t}`)}function bt(){u("git",["show","--name-only","--pretty=format:",...g()],{stdio:"inherit",silent:!0,triggerExit:!0})}import{spawn as me}from"node:child_process";async function $t(){let t=y("Error: missing valid target ref argument");await A();let e=await et(t);await l(["commit",`--fixup=${e}`]);let i=(await F(`${e}^`,"HEAD",!0)).map(n=>le(n));(await Promise.all(i)).every(Boolean)||(console.warn("Rebase conflict would occur!"),await s("git reset HEAD~1 --soft"),process.exit(1)),await s(`git rebase -i --autosquash ${e}^`)}async function le(t){let{stdout:e}=await s(`git diff ${t}^ ${t}`),{promise:o,resolve:i,reject:r}=Promise.withResolvers(),n=me("git",["apply","--check","--3way","-q"],{stdio:["pipe","ignore","ignore"]});return n.on("close",a=>{i(a===0)}),n.on("error",r),n.stdin.write(e),n.stdin.end(),o}import{readFile as fe,writeFile as pe}from"node:fs/promises";async function xt(){let t=await D("info/exclude");await H(t);let e=await L(t),o=g();o.filter(a=>!e.includes(a)).forEach(a=>console.info(`No match: '${a}'`));let r=(await fe(t,"utf-8")).split(/\r?\n/).map(a=>a.trim()),n=r.filter(a=>!o.includes(a));await pe(t,n.join(`
|
|
6
|
-
`)),console.info(`
|
|
7
|
-
Removed ${r.length-n.length} entries from ${t}`)}import{render as ge}from"ink";import{jsx as de}from"react/jsx-runtime";async function Pt(){let t=(async()=>{let r=await ue();return await we(r)})();ge(de(c,{msg:"Fetching repo languages",promise:t}));let e=await t,o=Object.values(e).reduce((r,n)=>r+n,0),i=Object.entries(e).map(([r,n])=>`${r}: ${Math.round(n/o*100)}% - ${he(n)}`);console.log(i.join(`
|
|
8
|
-
`))}async function ue(){let{stdout:t}=await s("gh repo view --json nameWithOwner -q .nameWithOwner");return t.trim()}async function we(t){let{stdout:e}=await s(`gh api repos/${t}/languages`);return JSON.parse(e.trim())}function he(t,e=2){if(t===0)return"0 B";let o=1024,i=e<0?0:e,r=["B","KB","MB","GB","TB","PB","EB","ZB","YB"],n=Math.floor(Math.log(t)/Math.log(o));return`${parseFloat((t/Math.pow(o,n)).toFixed(i))} ${r[n]}`}async function Rt(){let t=await Q();console.log(t)}import{render as At}from"ink";function Ct(){return new Date().toLocaleString("en-US",{year:"2-digit",month:"2-digit",day:"2-digit",hour:"numeric",minute:"2-digit",hour12:!0}).replace(",","")}async function v(){let t=Ct();await s("git add -A"),await l(["commit","-m",`"WIP ${t}"`])}import{nanoid as ye}from"nanoid";import be from"chalk";import{jsx as vt}from"react/jsx-runtime";async function Dt(){let{remote:t,branch:e}=await k(),o=process.argv[3]??e;await x()||await v(),await tt(o,"HEAD")||(console.warn("Conflicts would occur. Not safe to rebranch."),process.exit(1));let i=await h(),r=(async()=>{await E(t,o)})();At(vt(c,{msg:`Updating branch ${o}`,promise:r})),await r;let n=`${ye(8)}`,a=(async()=>{await s(`git switch ${i}`),await s(`git branch ${n}`),await s(`git switch ${o}`),await s(`git branch -D ${i}`),await s(`git switch -c ${i}`);let m=await F(o,n,!0);for(let f of m)try{await s(`git cherry-pick ${f}`)}catch{let b=be.yellow(`git branch -D ${i} && git branch -m ${n} ${i}`);throw new Error(`\u274C cherry pick failed at ${f}
|
|
9
|
-
Run this to restore backup branch
|
|
10
|
-
> ${b}`)}await s(`git branch -D ${n}`)})();At(vt(c,{msg:`Rewriting branch ${i}`,promise:a})),await a,console.info(`Successfully recreated branch ${i} from ${o}`)}import{execSync as $e}from"node:child_process";async function St(){R();let t=xe(process.argv[3]),{default:e}=await import("clipboardy");await e.write(t),console.log(`'${t}' copied to clipboard`)}function xe(t="HEAD"){return $e(`git rev-parse --short ${t}`,{encoding:"utf-8"}).trim()}import{render as Pe}from"ink";import{jsx as Re}from"react/jsx-runtime";async function Et(){let{remote:t,branch:e}=await K();d(t),await C();let o=W(t,e);Pe(Re(c,{msg:`Fetching branch ${e} from ${t}`,promise:o})),await o,await l(["switch","-c",e,"--track",`${t}/${e}`])}async function Y(){let{stdout:t}=await s("npm list -g bisgit --depth=0 --json");return JSON.parse(t.trim()).dependencies.bisgit.version}async function kt(){console.info(await Y())}async function Ce(){let{code:t}=await w("npm",["outdated","-g","bisgit"]);return t===0}async function Ae(){let{stdout:t}=await s("npm view bisgit version",{encoding:"utf8"});return t.trim()}async function Bt(){if(await Ce())return console.info("Already up to date");console.info(`Updating: ${await Y()} -> ${await Ae()}`),u("npm",["i","-g","bisgit@latest"])}async function qt(){let[t,e]=await Promise.allSettled([await De("user.name"),await ve()]);t.status==="fulfilled"&&console.info(`Git: ${t.value}`),e.status==="fulfilled"&&console.info(`Github: ${e.value}`)}async function ve(){let{stdout:t}=await s("gh api user --jq .login");return t.trim()}async function De(t){let{stdout:e}=await s(`git config ${t}`);return e.trim()}import{render as Ge}from"ink";import{Text as Se,useApp as Ee,useInput as ke}from"ink";import{useState as Be}from"react";import{Fragment as We,jsx as qe,jsxs as Wt}from"react/jsx-runtime";function Gt({prompt:t,msg:e,onConfirm:o}){let[i,r]=Be(""),{exit:n}=Ee();return ke((a,m)=>{if(i)return;let f=a.toLowerCase();f==="y"?r("Yes"):(f==="n"||m.return)&&(r("No"),n())}),Wt(We,{children:[Wt(Se,{children:[t," [y/N] ",i]}),i==="Yes"&&qe(c,{msg:e,promise:o()})]})}import{jsx as Fe}from"react/jsx-runtime";async function Ft(){if(rt(),await x())return console.log("Already a clean working tree");let t=await G();if(!t)throw new Error("git status cmd failed");let e=new J(t);e.displayReport(),Ge(Fe(Gt,{prompt:"Approve these changes?",msg:"Running git reset && git clean",onConfirm:async()=>(await s("git reset --hard"),await s("git clean -f"),`Altered ${e.willDelete.length+e.willRecreate.length+e.willRevert.length} files.`)}))}var J=class{willDelete=[];willRecreate=[];willRevert=[];constructor(e){e.forEach(o=>{if(o){let i=o.slice(0,2).trim(),r=o.slice(3);i.includes("?")||i.includes("A")?this.willDelete.push(r):i==="DU"||i==="D"?this.willRecreate.push(r):this.willRevert.push(r)}}),this.willDelete.sort(),this.willRecreate.sort(),this.willRevert.sort()}displayReport(){this.displaySection("Will Recreate:",this.willRecreate),this.displaySection("Will Revert:",this.willRevert),this.displaySection("Will Delete:",this.willDelete)}displaySection(e,o){o.length&&console.log(`${e}
|
|
11
|
-
${o.map(i=>`- ${i}
|
|
12
|
-
`)}`)}};import{nanoid as He}from"nanoid";import{render as Me}from"ink";import{jsx as Ne}from"react/jsx-runtime";async function Ht(){await M();let t=process.argv[3]??"origin";d(t);let e=await h();it(e);let o="",r=(async()=>{let n=[s(`git fetch ${t} ${e} --force`)];await x()||n.push(v()),await Promise.all(n);let[a,m]=await Promise.all([B(t,e),q(t,e)]);if(!(a===0&&m===0)){if(a!==0){let f=`${e}-${He(8)}`;o=`Created branch ${f}`,await s(`git branch ${f}`)}await s(`git reset --hard ${t}/${e}`)}})();Me(Ne(c,{msg:`Force pull reseting ${e} -> ${t}`,promise:r})),await r,o&&console.info(o)}var Mt={abort:st,amend:at,autoprune:pt,backmerge:mt,churn:gt,coauthor:ut,"code-review":wt,conflict:ht,continue:dt,exclude:yt,files:bt,fixup:$t,include:xt,languages:Pt,pwd:Rt,rebranch:Dt,"remote-default":ft,savepoint:v,sha:St,track:Et,update:Bt,"--version":kt,whoami:qt,wipe:Ft,yank:Ht};async function zi(t){return await Mt[t]?.(),Object.hasOwn(Mt,t)}export{zi as runCommand};
|
|
@@ -1,2 +0,0 @@
|
|
|
1
|
-
#!/usr/bin/env node
|
|
2
|
-
import{a as we,b as Se,c as Me,d as Pe,j as Ce,o as Ee,t as Ae,u as ke,v as Be}from"./chunk-DLHREY2Q.js";import{b as C}from"./chunk-R7DNYVL6.js";import{render as Gr}from"ink";import{createContext as Ht,useContext as Vt,useState as $t}from"react";import{jsx as _t}from"react/jsx-runtime";var Re=Ht(void 0),Le=({children:e})=>{let[t,o]=$t({}),r=(i,s)=>{o(c=>({...c,[i]:s}))},n=i=>{let s=Array.isArray(i)?i:[i];o(c=>{let l={...c};for(let a of s)delete l[a];return l})};return _t(Re.Provider,{value:{keybindings:t,setKeybinding:r,removeKeybinding:n},children:e})},M=()=>{let e=Vt(Re);if(!e)throw new Error("useKeybindings must be used within a KeybindingsProvider");return e};import{useEffect as Or}from"react";import{Text as Wr,Box as It,useInput as Ir}from"ink";import{useStdout as Ut}from"ink";function f(){let{stdout:e}=Ut(),t=e.rows-2,o=Math.floor(t/2),r=e.columns>100?96:e.columns-4;return{width:e.columns,height:e.rows,sectionHeight:t,sectionHalfHeight:o,modalWidth:r}}import{Box as Fe,Text as zt}from"ink";import{useState as ee,createContext as Yt,useContext as qt}from"react";import{jsx as Qt}from"react/jsx-runtime";var D={Files:"Files","File Meta":"Files",Status:"Status",Log:"Log","Commit Details":"Log",Branches:"Branches",Worktrees:"Branches",Remotes:"Branches"},H=Object.keys(D),Ne=Yt({});function De({children:e}){let[t,o]=ee(!1),r=()=>o(!0),n=()=>o(!1),[i,s]=ee("Status"),[c,l]=ee("Status"),a=u=>{t||(l(u),s(D[u]))},m=u=>{t||(s(u),l(u))},p=()=>{t||l(u=>{let v=H.at((H.indexOf(u)+1)%H.length);return s(D[v]),v})},b=()=>{t||l(u=>{let v=H.at(H.indexOf(u)-1);return s(D[v]),v})};return Qt(Ne.Provider,{value:{activeGroup:i,setActiveGroup:m,activeSection:c,setActiveSection:a,prevSection:b,nextSection:p,isLocked:t,lock:r,unlock:n},children:e})}function w(){let e=qt(Ne);if(Object.keys(e).length===0)throw new Error("useNavContext must be used within a NavProvider");return e}import{memo as Jt}from"react";import{jsx as te}from"react/jsx-runtime";var Xt=Array.from(new Set(Object.values(D))),Oe=Jt(({activeGroup:e})=>te(Fe,{width:"100%",flexDirection:"row",height:1,children:Xt.map((t,o)=>te(Fe,{borderStyle:"classic",borderLeft:!!o,borderRight:!1,borderTop:!1,borderBottom:!1,children:te(zt,{color:t===e?"cyan":"",children:` [${o+1}]: ${t} `})},t))}),(e,t)=>e.activeGroup===t.activeGroup);import{Text as Lr,useInput as Nr}from"ink";import{Box as no,Text as io}from"ink";import{Box as We,measureElement as Zt,Text as Ie}from"ink";import{useMemo as jt,useRef as eo}from"react";import{Fragment as to,jsx as $,jsxs as oo}from"react/jsx-runtime";function S(e){let{title:t,footer:o,children:r,width:n,height:i,innerHeight:s,isModal:c,backgroundColor:l}=e,{width:a}=f(),m=n??a,p=eo(null),b=jt(()=>{if(!p.current)return 0;let{width:h}=Zt(p.current);return h},[p.current]),u=b-8,v={...e,height:s,width:b},{activeSection:x}=w(),d=x===t||c?"cyan":"";return $(We,{ref:p,flexDirection:"column",height:i,width:m,backgroundColor:l,children:b!==0&&oo(to,{children:[t&&$(Ie,{color:d,wrap:"truncate-end",children:"\u256D\u2500\u2500\u2500\u2500 "+t+" "+"\u2500".repeat(u-t.length)+"\u256E"}),$(We,{flexDirection:"column",...v,borderStyle:"round",borderColor:d,borderTop:!t,borderBottom:!o,paddingLeft:1,children:r}),o&&$(Ie,{color:d,wrap:"truncate-end",children:"\u2570"+"\u2500".repeat(u-o.length)+" "+o+" \u2500\u2500\u2500\u2500\u256F"})]})})}import{useEffect as ro,useState as oe}from"react";function y(e,t=[]){let[o,r]=oe(void 0),[n,i]=oe(!1),[s,c]=oe(0),l=()=>c(a=>a+1);return ro(()=>{let a=!1;return i(!1),r(void 0),e().then(m=>{a||(r(m),i(!0))}).catch(()=>{a||i(!0)}),()=>{a=!0}},[s,...t]),{value:o,resolved:n,refresh:l}}import{jsx as re}from"react/jsx-runtime";function Ke(){let{width:e,sectionHeight:t}=f(),{value:o}=y(Ee);return re(no,{width:e,height:t,children:re(S,{title:"Status",innerHeight:t-1,children:o?.map(r=>re(io,{children:r},r))})})}import{useEffect as gt,useState as ht}from"react";import{Box as tr,useInput as or}from"ink";import{Text as so}from"ink";import{jsx as Ge}from"react/jsx-runtime";function He({sha:e}){let{resolved:t,value:o}=y(()=>co(e),[e]);return Ge(S,{title:"Commit Details",width:"50%",height:"100%",children:e&&t&&Ge(so,{children:o})})}async function co(e){if(!e)return;let{stdout:t}=await C(`git show ${e} --name-only`);return t.trim()}import{useCallback as To,useEffect as wo}from"react";import{Box as U,Text as se,useInput as So}from"ink";import{useEffect as Ve,useRef as ao,useState as ne}from"react";function _(e,t){let[o,r]=ne(0),[n,i]=ne(0),s=ao(void 0),[c,l]=ne(0),a=()=>l(d=>d+1);Ve(()=>{let d=0;if(s.current!==void 0){let g=e.findIndex(T=>T===s.current);g>=0&&(d=g)}i(d);let h=Math.min(Math.max(d-Math.floor(t/2),0),Math.max(0,e.length-t));r(h)},[e,t,c]),Ve(()=>{s.current=e[n]},[n,e]);let m=Math.max(0,e.length-t),p=e.length-1,b=()=>{i(d=>{let h=Math.min(d+1,p);return r(g=>h>=g+t?Math.min(g+1,m):g),h})},u=()=>{i(d=>{let h=Math.max(d-1,0);return r(g=>h<g?Math.max(g-1,0):g),h})},v=e.slice(o,o+t),x=e[n];return{scrollDown:b,scrollUp:u,outList:v,selectedIndex:n,selectedValue:x,refresh:a}}import{useInput as lo}from"ink";import{createContext as uo,useContext as mo}from"react";import{useEffect as Ye,useState as $e}from"react";import{jsx as po}from"react/jsx-runtime";var _e="\u2190",Ue="\u2192",qe=uo({});function Qe({children:e}){let[t,o]=$e("truncate-end"),[r,n]=$e(0),{removeKeybinding:i,setKeybinding:s}=M();return Ye(()=>(r&&(t==="truncate-start"?s(_e,"pan left"):s(Ue,"pan right")),()=>{i([Ue,_e])}),[t,r]),lo((c,l)=>{r&&(l.rightArrow?o("truncate-start"):l.leftArrow&&o("truncate-end"))}),po(qe.Provider,{value:{mode:t,_isToggleable:r,_setIsToggleable:n},children:e})}function F(){let e=mo(qe);if(Object.keys(e).length===0)throw new Error("useTruncationMode must be used within an TruncationModeProvider");let{mode:t,_setIsToggleable:o}=e;return Ye(()=>(o(r=>r+1),()=>o(r=>r-1)),[]),{mode:t}}import{createContext as fo,useContext as go,useEffect as ze,useMemo as ho,useRef as Je,useState as Xe}from"react";import{useInput as xo}from"ink";import{jsx as bo}from"react/jsx-runtime";var Ze=fo({});function je({children:e}){let{activeSection:t}=w(),{sectionHeight:o}=f(),r=o-3,{value:{tree:n,files:i}={},resolved:s}=y(vo),c=Je(new Set),l=Je(new Set(["."])),[a,m]=Xe();ze(()=>{n&&i&&(c.current=new Set(Object.keys(n)),m(i[0]))},[s]);let p=ho(()=>i?.filter(x=>c.current.has(x))??[],[c.current.size,s]),[b,u]=Xe(0),v=()=>u(x=>x+1);return xo((x,d)=>{if(t!=="Files"||!n||!a)return;if(d.upArrow){let g=Math.max(0,p.indexOf(a)-1);m(p[g]),v()}else if(d.downArrow){let g=Math.min(p.length-1,p.indexOf(a)+1);m(p[g]),v()}let h;if(d.leftArrow?h="collapse":d.rightArrow&&(h="expand"),h){let g=ie(n,a);if(!g)return;for(let T of Object.keys(g)){let E=`${a}/${T}`;h==="expand"?c.current.add(E):c.current.delete(E)}v()}}),ze(()=>{if(l.current.clear(),l.current.add("."),p.length<=r){p.forEach(G=>l.current.add(G));return}let x=a?yo(a):[],d=Math.max(1,r-x.length),h=p.filter(G=>!x.includes(G)),g=a?h.indexOf(a):0,T=Math.max(0,g-Math.floor(d/2)),E=T+d;E>h.length&&(E=h.length,T=Math.max(0,E-d));let Gt=h.slice(T,E);[...x,...Gt].forEach(G=>{l.current.add(G)})},[p.length,a,r]),bo(Ze.Provider,{value:{version:b,tree:n,visibleFiles:l.current,expandedFiles:c.current,selectedFile:a},children:e})}function A(){let e=go(Ze);if(Object.keys(e).length===0)throw new Error("useTreeNavigation must be used within an TreeNavigationProvider");return e}var ie=(e,t)=>{let o=t.split("/"),r=e;for(let n of o)r=r[n];return r},vo=async()=>{let e=await Se(),t=Me(e);return{tree:e,files:t}},yo=e=>{let t=e.split("/"),o=[];for(let r=1;r<t.length;r++)o.push(t.slice(0,r).join("/"));return o};import{jsx as R,jsxs as Mo}from"react/jsx-runtime";function et({setSha:e,mode:t}){let{label:o,value:r}=t,{sectionHeight:n}=f(),{activeSection:i,isLocked:s}=w(),{selectedFile:c}=A(),{mode:l}=F(),a=To(()=>r.length&&c?r(c):r(),[r,c]),{resolved:m,value:p=[]}=y(a,[o]),{scrollUp:b,scrollDown:u,outList:v,selectedValue:x}=_(p,n-2),d=x?.sha;return wo(()=>{e(d)},[d]),So((h,g)=>{i!=="Log"||s||(g.upArrow?b():g.downArrow&&u())}),R(S,{overflowY:"hidden",height:"100%",title:"Log",width:"50%",children:m&&v.map(({sha:h,message:g},T)=>Mo(U,{flexDirection:"row",flexWrap:"nowrap",children:[R(U,{minWidth:2,children:h===d?R(se,{children:"> "}):null}),R(U,{minWidth:8,children:R(se,{color:"yellow",children:h})}),R(U,{children:R(se,{wrap:l,children:g})})]},`${h}-${t.label}${t.label==="reflog"?T:""}`))})}import{useState as Go}from"react";import{useInput as Ho}from"ink";import Vo from"ink-text-input";import{useInput as Po}from"ink";import{createContext as Co,useContext as Eo,useEffect as Ao,useState as ce}from"react";import{Fragment as ko,jsx as Ro,jsxs as Bo}from"react/jsx-runtime";var tt=Co({});function P(){let e=Eo(tt);if(Object.keys(e).length===0)throw new Error("useKeybindings must be used within a KeybindingsProvider");return e}function ot({children:e}){let[t,o]=ce(null),{lock:r,unlock:n}=w(),[i,s]=ce(!1),[c,l]=ce(),a=()=>{n(),s(!1),c?.(),o(null)},m=()=>{r(),s(!0)},p=()=>{i?a():m()},b=(x,d)=>{a(),l(d?.onClose),o(x)};Po((x,d)=>{d.escape&&a()});let{setKeybinding:u,removeKeybinding:v}=M();return Ao(()=>{if(i)return u("esc","close"),()=>{v("esc")}},[i]),Ro(tt.Provider,{value:{setModal:b,open:m,close:a,toggle:p,isOpen:i},children:Bo(ko,{children:[e,t]})})}import{Box as st}from"ink";import ae from"node:path";import le from"node:os";import Lo from"node:fs/promises";import{createContext as No,useContext as Do}from"react";import{jsx as Io}from"react/jsx-runtime";var rt=No({});function nt({children:e}){let{value:t}=y(Fo);return Io(rt.Provider,{value:{bgColor:t},children:e})}var it=()=>{let e=Do(rt);if(Object.keys(e).length===0)throw new Error("useKeybindings must be used within a KeybindingsProvider");return e};async function Fo(){if(!(!process.stdin.isTTY||!process.stdout.isTTY))return process.env.TERM_PROGRAM==="vscode"?await Wo():new Promise(e=>{let t="\x1B]11;?\x07",o="",r,n=s=>{clearTimeout(r),process.stdin.off("data",i);try{process.stdin.setRawMode(!1),process.stdin.pause()}catch{}e(s)},i=s=>{o+=s.toString("ascii");let c=o.match(/\x1b]11;rgb:([0-9a-fA-F]{1,4})\/([0-9a-fA-F]{1,4})\/([0-9a-fA-F]{1,4})\x07/);if(c){let l=Oo(c[1],c[2],c[3]);n(l)}};try{process.stdin.setRawMode(!0),process.stdin.resume(),process.stdin.on("data",i),r=setTimeout(()=>n(void 0),150),process.stdout.write(t)}catch{n(void 0)}})}function Oo(e,t,o){return"#"+[e,t,o].map(r=>Math.round(parseInt(r,16)*255/((1<<4*r.length)-1)).toString(16).padStart(2,"0")).join("")}async function Wo(){let e="Code";process.env.TERM_PROGRAM_VERSION?.includes("insider")&&(e="Code - Insiders");let t;switch(process.platform){case"win32":t=ae.join(le.homedir(),"AppData","Roaming",e,"User","settings.json");break;case"linux":t=ae.join(le.homedir(),".config",e,"User","settings.json");break;case"darwin":t=ae.join(le.homedir(),"Library","Application Support",e,"User","settings.json");break;default:return}if(!await we(t))return;let o=await Lo.readFile(t,{encoding:"utf-8"}),{default:r}=await import("strip-json-comments"),n=r(o).replace(/,\s*(\}|\])/g,"$1");switch(JSON.parse(n)["workbench.colorTheme"]){case"Default Dark Modern":case void 0:return"#181818";case"Visual Studio Dark":case"Default Dark+":return"#1e1e1e";case"Default High Contrast":return"#000000";case"Default Light+":case"Visual Studio Light":case"Default High Contrast Light":return"#ffffff";case"Default Light Modern":return"#f8f8f8";default:return}}import{jsx as de}from"react/jsx-runtime";var Ko=36;function O({children:e,title:t,width:o=Ko}){let r=f(),{width:n}=r,i=Math.floor((n-o)/2),{bgColor:s}=it();return de(st,{...r,position:"absolute",children:de(st,{alignSelf:"center",marginLeft:i,marginRight:i,children:de(S,{title:t,width:o,isModal:!0,backgroundColor:s,children:e})})})}import{jsx as ct}from"react/jsx-runtime";function at({title:e,handleSubmit:t}){let[o,r]=Go(""),{isOpen:n,close:i}=P();return Ho((s,c)=>{c.return&&n&&(t(o),r(""),i())}),n&&ct(O,{title:e,children:ct(Vo,{value:o,onChange:r})})}import{useInput as $o}from"ink";import _o from"ink-select-input";import{jsx as lt}from"react/jsx-runtime";function dt({title:e,handleSubmit:t,options:o,initialIndex:r,width:n}){let{isOpen:i,close:s}=P();return $o((c,l)=>{l.return&&i&&s()}),i&<(O,{title:e,width:n,children:lt(_o,{items:o,onSelect:t,initialIndex:r})})}var I=[{value:()=>W("git log --oneline -n 400"),label:"log"},{value:()=>W("git log --oneline --tags --no-walk --decorate"),label:"tags"},{value:()=>W("git reflog"),label:"reflog"},{value:()=>W('git log --oneline --author="$(git config user.name)"'),label:"my commits"},{value:e=>W(`git log --oneline "${e}"`),label:"file"},{value:e=>W(`git blame --follow "${e}"`),label:"blame"}];async function W(e){let{stdout:t}=await C(e);return t.trim().split(/\r?\n/).map(Uo)}function Uo(e){let t=e.indexOf(" ");return{sha:e.slice(0,t),message:e.slice(t+1)}}import{createContext as Yo,useContext as qo,useEffect as Qo,useState as zo}from"react";import{jsx as Jo}from"react/jsx-runtime";var ut=Yo({});function mt({children:e}){let[t,o]=zo(),r=(n,...i)=>{n(...i).catch(s=>o(s.stderr??s.message))};return Qo(()=>{if(t)throw new Error(t)},[t]),Jo(ut.Provider,{value:{attempt:r,err:t},children:e})}function Y(){let e=qo(ut);if(Object.keys(e).length===0)throw new Error("useErrorCatcher must be used within an ErrorCatcherProvider");return e}import{createContext as Xo,useContext as Zo,useState as jo}from"react";import{jsx as er}from"react/jsx-runtime";var pt=Xo({});function ft({children:e}){let[t,o]=jo(),r=i=>o(i),n=i=>{let s=t===i;return s&&o(void 0),s};return er(pt.Provider,{value:{sendMessage:r,receiveMessage:n},children:e})}function q(){let e=Zo(pt);if(Object.keys(e).length===0)throw new Error("useMessaging must be used within an MessagingProvider");return e}import{jsx as ue,jsxs as rr}from"react/jsx-runtime";function me(){let{setModal:e,toggle:t}=P(),[o,r]=ht(),{width:n,sectionHeight:i}=f(),[s,c]=ht(I[0]),{receiveMessage:l}=q();gt(()=>{l("log-file")&&c(I[4]),l("blame-file")&&c(I[5])},[]);let{isLocked:a}=w(),{attempt:m}=Y();or((u,v)=>{u==="m"&&(e(ue(dt,{options:I,title:"Log Type",handleSubmit:x=>c(x),initialIndex:I.indexOf(s)})),t()),!a&&u==="c"&&o&&m(()=>C(`git checkout ${o}`))});let{setKeybinding:p,removeKeybinding:b}=M();return gt(()=>(p("c","checkout"),p("m","mode"),()=>{b(["c","m"])}),[]),rr(tr,{width:n,height:i,children:[ue(et,{setSha:r,mode:s}),ue(He,{sha:o})]})}import{useEffect as mr}from"react";import{Box as wt}from"ink";import{Box as pe,Text as xt,useInput as nr}from"ink";import{useEffect as ir}from"react";import{Fragment as cr,jsx as L,jsxs as sr}from"react/jsx-runtime";function vt(){let{activeSection:e,isLocked:t}=w(),{sectionHalfHeight:o}=f(),{value:r=[],refresh:n}=y(Ae),{value:i,refresh:s}=y(Pe),{setKeybinding:c,removeKeybinding:l}=M();ir(()=>{if(e==="Branches")return c("s","switch"),()=>{l("s")}},[e]);let{outList:a,selectedValue:m,scrollUp:p,scrollDown:b}=_(r,o-2),u=async T=>{await C(`git branch ${T}`),n()},v=async()=>{await C(`git branch -D ${m}`),n()},x=async()=>{await C(`git switch ${m}`),n(),s()},{setModal:d,open:h}=P(),{attempt:g}=Y();return nr((T,E)=>{e!=="Branches"||t||(E.upArrow?p():E.downArrow?b():T==="c"?(d(L(at,{title:"Name your new branch",handleSubmit:u})),h()):T==="d"?g(v):T==="s"&&g(x))}),L(cr,{children:L(S,{width:"100%",title:"Branches",innerHeight:o-1,children:a.map(T=>sr(pe,{flexDirection:"row",flexWrap:"nowrap",children:[L(pe,{minWidth:2,children:T===m?L(xt,{children:"> "}):null}),L(pe,{minWidth:8,children:L(xt,{color:T===i?"magenta":void 0,children:T})})]},T))})})}import{Text as fe}from"ink";import{Fragment as ar}from"react";import{jsx as ge,jsxs as yt}from"react/jsx-runtime";function bt(){let{value:e}=y(ke);return ge(S,{width:"50%",title:"Remotes",children:e?.map(({url:t,name:o,type:r})=>yt(ar,{children:[yt(fe,{wrap:"truncate-end",children:[o," (",r,")"]}),ge(fe,{wrap:"truncate-end",children:t}),ge(fe,{children:" "})]},`${o}-${r}`))})}import{Text as he}from"ink";import{Fragment as lr}from"react";import{jsx as Q,jsxs as ur}from"react/jsx-runtime";function Tt(){let{sectionHalfHeight:e}=f(),{value:t}=y(dr);return Q(S,{width:"100%",title:"Worktrees",innerHeight:e-1,children:t?.map(({branch:o,detached:r,path:n,head:i})=>ur(lr,{children:[Q(he,{children:r?i:o}),Q(he,{children:n}),Q(he,{children:" "})]},n))})}async function dr(){let{stdout:e}=await C("git worktree list --porcelain"),t=e.trim().split(/\r?\n/),o=[],r=[];for(let n of t)n.trim()===""?r.length&&(o.push(r),r=[]):r.push(n);return r.length&&o.push(r),o.map(n=>{let i={detached:!1};for(let s of n)s.startsWith("worktree ")?i.path=s.slice(9):s.startsWith("HEAD ")?i.head=s.slice(5):s.startsWith("branch ")?i.branch=s.slice(7):s==="detached"&&(i.detached=!0);return i})}import{Fragment as pr,jsx as z,jsxs as St}from"react/jsx-runtime";function Mt(){let{width:e,sectionHeight:t}=f(),{setKeybinding:o,removeKeybinding:r}=M();return mr(()=>(o("c","create"),o("d","delete"),()=>{r(["c","d"])}),[]),z(pr,{children:St(wt,{width:e,height:t,children:[St(wt,{flexDirection:"column",width:"50%",height:t,children:[z(vt,{}),z(Tt,{})]}),z(bt,{})]})})}import{Component as fr,useEffect as gr}from"react";import{Text as Pt}from"ink";import{jsx as xe}from"react/jsx-runtime";function J({title:e,message:t,width:o}){let{isOpen:r}=P(),{mode:n}=F();return r&&xe(O,{title:e,width:o,children:Array.isArray(t)?t.map(i=>xe(Pt,{wrap:n,children:i})):xe(Pt,{children:t})})}import{jsx as Ct}from"react/jsx-runtime";var X=class extends fr{state={hasError:!1};constructor(t){super(t)}static getDerivedStateFromError(t){return{hasError:!0,message:t?.stderr??t?.message}}dismissError(){this.setState({hasError:!1})}componentDidCatch(t,o){let r=t?.stderr??t?.message;this.setState({hasError:!0,message:r})}render(){return this.state.hasError?Ct(hr,{dismissError:this.dismissError.bind(this),message:this.state.message,children:this.props.children}):this.props.children}};function hr({message:e,dismissError:t,children:o}){let{setModal:r,open:n}=P(),{modalWidth:i}=f();return gr(()=>{if(e){let s=e.split(/\r?\n/).map(c=>c.trim());r(Ct(J,{title:"Error",message:s,width:i}),{onClose:t}),n()}},[e]),o}import{Box as Ar,useInput as kr}from"ink";import{useEffect as Br}from"react";import{Box as ve,Text as Bt}from"ink";import ye from"node:path";import{memo as xr,useEffect as Et,useState as vr}from"react";import{Fragment as wr,jsx as B,jsxs as be}from"react/jsx-runtime";var yr="\u2190",br="\u2191",Rt="\u2192",Lt="\u2193",At=`${yr}${br}${Lt}${Rt}`;function Nt(){let{sectionHeight:e}=f(),{activeSection:t}=w(),{setKeybinding:o,removeKeybinding:r}=M();Et(()=>{if(t==="Files")return o(At,"NAVIGATION_KEY"),()=>r(At)},[t]);let{tree:n,version:i}=A(),{value:s}=y(Ce),[,c]=vr(0);return Et(()=>{c(l=>l+1)},[i]),B(S,{title:"Files",innerHeight:e,width:"50%",children:s&&n&&B(Dt,{name:ye.basename(s),contents:n,depth:0,fp:s})})}function Dt(e){let{contents:t,depth:o,fp:r,rp:n="."}=e,i=Object.keys(t).length===0,{selectedFile:s,visibleFiles:c,expandedFiles:l}=A(),a=n===s;if(!c.has(n))return null;let m=o+2;if(i)return B(kt,{...e,depth:m,isSelected:a},r);let p=`${n}/`,b=Tr(l,u=>u.startsWith(p))||o===0;return be(wr,{children:[be(ve,{flexDirection:"row",flexWrap:"nowrap",children:[B(ve,{width:2,children:B(Bt,{children:b?Lt:Rt})}),B(ve,{width:"100%",children:B(kt,{...e,isSelected:a})})]}),b&&Object.entries(t).map(([u,v])=>{let x=ye.join(n,u),d=ye.join(r,u);return B(Dt,{name:u,contents:v,depth:m,rp:x,fp:d})})]})}var kt=xr(({name:e,depth:t,isSelected:o})=>be(Bt,{wrap:"truncate-end",color:o?"yellow":void 0,children:["\xA0".repeat(t),e]}),(e,t)=>e.isSelected===t.isSelected&&e.fp===t.fp);function Tr(e,t){for(let o of e)if(t(o))return!0;return!1}import{Box as Sr,measureElement as Mr,Text as K}from"ink";import{useMemo as Pr,useRef as Cr}from"react";import{jsx as Z,jsxs as V}from"react/jsx-runtime";function Ft(){let{sectionHeight:e}=f(),{selectedFile:t}=A(),o=Cr(null),r=Pr(()=>{if(!o.current)return 0;let{width:m}=Mr(o.current);return m-4},[o.current]),{value:n={}}=y(()=>Be(t),[t]),{authorDate:i,authorEmail:s,authorName:c,shortHash:l}=n,{mode:a}=F();return Z(Sr,{width:"50%",height:e,ref:o,children:V(S,{width:"100%",title:"File Meta",innerHeight:e-1,children:[t&&Z(Er,{width:r,text:t}),l&&V(K,{wrap:a,children:["Last Commit: ",l]}),c&&V(K,{wrap:a,children:["Edited By: ",c]}),i&&V(K,{wrap:a,children:["Edit On: ",i]}),s&&V(K,{wrap:a,children:["Contact: ",s]})]})})}function Er({width:e,text:t}){if(t.length>e)return Z(K,{children:t});let o=Math.max(0,e-t.length-2),r=Math.floor(o/2),n=o-r,i=`${"=".repeat(r)} ${t} ${"=".repeat(n)}`;return Z(K,{children:i})}import{jsx as Te,jsxs as Rr}from"react/jsx-runtime";function Ot(){let{width:e,sectionHeight:t}=f(),{selectedFile:o,tree:r}=A(),{setKeybinding:n,removeKeybinding:i}=M();Br(()=>(n("l","log"),n("b","blame"),()=>{i(["l","b"])}),[]);let{setActiveGroup:s}=w(),{sendMessage:c}=q(),{setModal:l,open:a}=P();return kr((m,p)=>{m==="l"?(c("log-file"),s("Log")):m==="b"&&(Object.keys(ie(r??{},o)).length===0?(c("blame-file"),s("Log")):(l(Te(J,{title:"Sorry",message:["Can't blame a folder."]})),a()))}),Rr(Ar,{width:e,height:t,children:[Te(Nt,{}),Te(Ft,{})]})}import{jsx as k,jsxs as Fr}from"react/jsx-runtime";function Wt(){let{prevSection:e,nextSection:t,activeGroup:o,setActiveGroup:r}=w();return Nr((n,i)=>{i.tab&&i.shift?e():i.tab?t():n==="1"?r("Files"):n==="2"?r("Status"):n==="3"?r("Log"):n==="4"&&r("Branches")}),k(X,{children:k(mt,{children:k(ft,{children:k(je,{children:k(Dr,{group:o})})})})})}function Dr({group:e}){switch(e){case"Files":return k(Ot,{});case"Status":return k(Ke,{});case"Log":return k(me,{});case"Branches":return k(Mt,{});default:return Fr(Lr,{children:['Error, unknown page "',e,'"']})}}import{jsx as j,jsxs as Kr}from"react/jsx-runtime";function Kt(){let{isLocked:e,activeGroup:t}=w();Ir((i,s)=>{e||i.toLowerCase()==="q"&&process.exit()});let o=f(),{keybindings:r,setKeybinding:n}=M();return Or(()=>n("q","quit"),[]),Kr(It,{flexDirection:"column",...o,children:[j(Oe,{activeGroup:t}),j(Wt,{}),j(It,{flexDirection:"row",children:Object.entries(r).map(([i,s])=>j(Wr,{children:`[${i}]: ${s} `},s))})]})}import{jsx as N}from"react/jsx-runtime";function Hr(){return N(nt,{children:N(De,{children:N(Le,{children:N(Qe,{children:N(ot,{children:N(Kt,{})})})})})})}var da=()=>Gr(N(Hr,{}),{patchConsole:!0});export{da as renderApp};
|